Chinaunix首页 | 论坛 | 博客
  • 博客访问: 58355
  • 博文数量: 11
  • 博客积分: 410
  • 博客等级: 一等列兵
  • 技术积分: 114
  • 用 户 组: 普通用户
  • 注册时间: 2010-01-23 01:10
文章分类

全部博文(11)

文章存档

2013年(1)

2012年(1)

2011年(3)

2010年(6)

我的朋友

分类: C/C++

2011-04-20 20:41:28

 题记: 要知道,并不是只有初学者才会犯错。(shiroki的至理名言)最近发现了一些有意思的问题,值得memo一下。
先来看段代码:
#include
#include
#include
int main(int argc, char* argv[])
{
QApplication a(argc, argv);
QWebView* mw = new QWebView;
mw->show();
mw->load(QUrl("http://www.cuteqt.com/blog"));
return a.exec();
}

大家看得出这段代码中的问题吗? (呵呵,不要告诉我是cuteqt不能访问哦~)

这段代码ms十分标准, 非常符合笔者平时写Qt程序书写main函数的习惯, 孰料想竟然是个错误的习惯,而且问题很严重哦。 给个提示:在程序退出时会aborted。

如果还没想出来是什么问题,嘿嘿,没关系,看了下面的答案你就明白了。

在这段程序里QApplication实例创建在stack上,生命期是main的大括号内, 而mw则通过new创建在heap上, 在程序退出时才会被析构。 换句话说,mw的生存期长于application的生存期…..这可是Qt编程的大忌, 因为在Qt中所有的Paint Device都必须要在有QApplication实例的情况下创建和使用。 不过如果把这个程序写出来运行一下, 未必会出现我说的aborted的问题,  大多数代码类似的程序都能安全的运行(这也是为什么用了那么多年的Qt从来没有注意过这个问题, 并且养成了我错误的编程习惯。)。  这里的trick在于application退出时mw已经被关闭, mw中的所有Paint Device一般都不会被访问到了, 所以这个错误隐藏在很深的阴暗角落, 偷偷地嘲笑我们呢!

要想试验这个问题也很简单,把load的参数换成本地文件 test.html, 并把下面的内容写进test.html就能看到拉:


-----test.html-----


这个html里使用了下拉选单。 如果你运行程序并点开该选单,之后退出程序你就会看到Aborted错误提示,并打印出错误信息:“QWidget: Must construct a QApplication before a QPaintDevice”。

既然提出的问题,当然也要给出解决的方案。 有两种可行的方法避免该错误。 一个当然是纠正一下编程习惯,对mw不要用new的方式创建,改在stack上创建,如下代码:

#include
#include
#include
int main(int arg, char* argv[])
{
QApplication a(argc, argv);
QWebView mw;
mw.show();
mw.load(QUrl("http://www.cuteqt.com/blog"));
return a.exec();
}

另外还可以用Qt提供的API解决此问题, 想办法让mw在application之前clean up, 那就是用WA_DeleteOnClose属性。 该属性标示窗体会在close时被析构, 这样就保证不会留存在application析构之后了, 是个很好的办法。

代码如下:

#include
#include
#include
int main(int arg, char* argv[])
{
QApplication a(argc, argv);
QWebView* mw = new QWebView;
mw->show();
mw->setAttribute(Qt::WA_DeleteOnClose);
mw->load(QUrl("http://www.cuteqt.com/blog"));
return a.exec();
}
阅读(843) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~