Qt 程序退出时断言错误——_BLOCK_TYPE_IS_VALID(pHead->nBlockUse),由setAttribute(Qt::WA_DeleteOnClose)引起

最近在学习QT,自己仿写了一个简单的QT绘图程序,但是在退出时总是报错,断言错误:

报错主要问题在_BLOCK_TYPE_IS_VALID(pHead->nBlockUse),是在关闭窗口时报的错;

先前考虑是析构函数有问题,重写并且排查相关变量并未发现问题。

根据报错问题又推测栈调用出现内存溢出,寻找程序中所有的链表和栈调用。未果。

最后用了最笨的方法,将构造函数中所有变量挨个遍历,最终确定问题出在Qt的setAttribute(Qt::WA_DeleteOnClose)这行代码上。

网上查询原来Qt初始化时有两种构造方式:

                          当MainWindow   w;  w.show();时,是将窗口对象定义在栈上,

          当MainWindow  *w = new MainWindow ;时,则是定义在堆上。

而setAttribute函数原型为:void QWidget::setAttribute ( Qt::WidgetAttribute attribute, bool on = true )

Qt文档解释:Sets the attribute attribute on this widget if on is true; otherwise clears the attribute.

简单翻译就是:当on为true时,设置当前窗口的状态attribute为开启状态,其他则清除当前attribute状态。

而关于Qt::WA_DeleteOnClose,从字面意思来看不难猜测是在窗口关闭是调用delete来析构当前窗口指针,可以简单理解成在关闭窗口时执行了delete w;

但是当前窗口若定义在栈上,w是一个对象而不是指针,显然不正确,而报断言错误的原因则是当程序执行到这里是“w指针"不存在,故而报出断言错误。

 

 

 关于堆栈,可以参考:什么是堆栈?它们在哪?

 

 

总结:1、在Qt中有两种内存分配方式(基于C的,一定和C一样啦~),分别是堆分配和栈分配。

   2、当MainWindow   w;  w.show();时,是将窗口对象定义在栈上,

      当MainWindow  *w = new MainWindow ;时,则是定义在堆上。

     3、setAttribute()函数可以设置窗口状态属性,当窗口对象定义在堆上,需要开启Qt::WA_DeleteOnClose状态;而在定义在栈上,开启这个状态则会报错。

   4、写程序养成留下断言的习惯,可以及早发现程序错误,断言真的很好使。

     5、做程序开发,一定要亲力亲为,和大多数事情一样,没有捷径可走。脚踏实地做下去,会有令人欣慰的成果。

 

                                            2014年11月15日于秦皇岛

 

posted @ 2014-11-15 00:21  澄轶  阅读(1633)  评论(0编辑  收藏  举报