本文介绍的是分析Qt编程习惯 主窗体创建问题,要知道,并不是只有初学者才会犯错。最近发现了一些有意思的问题,值得memo一下。
先来看段代码:
- #include <QApplication>
- #include <QWebView>
- #include <QUrl>
- 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就能看到拉:
- <form>
- <select id="headertest">
- <option>Item1</option>
- <option>Item2</option>
- <option>Item3</option>
- </select>
- </form>
这个html里使用了下拉选单。如果你运行程序并点开该选单,之后退出程序你就会看到Aborted错误提示,并打印出错误信息:“QWidget: Must construct a QApplication before a QPaintDevice”。
既然提出的问题,当然也要给出解决的方案。 有两种可行的方法避免该错误。 一个当然是纠正一下编程习惯,对mw不要用new的方式创建,改在stack上创建,如下代码:
- #include <QApplication>
- #include <QWebView>
- #include <QUrl>
- 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 <QApplication>
- #include <QWebView>
- #include <QUrl>
- 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();
- }
发现问题和解决问题是件很有乐趣的事情,大家不要把时间都浪费在猜测上,要多动手多思考才能进步!
Qt Jambi也存在类似的问题,如果以程序启动的代码块去启动QApplication,在程序运行过程中,一些资源回收会报出Null指针错误,这些错误,通过debug,最终都会指向QWidget这个类。当把QApplication启动的执行程序移出main函数,问题迎刃而解。要多注意细节。
小结:分析Qt编程习惯 主窗体创建问题的内容介绍完了,希望本文对你有所帮助!更多内容请参考编辑推荐。