分析Qt編程習(xí)慣 主窗體創(chuàng)建問(wèn)題
本文介紹的是分析Qt編程習(xí)慣 主窗體創(chuàng)建問(wèn)題,要知道,并不是只有初學(xué)者才會(huì)犯錯(cuò)。最近發(fā)現(xiàn)了一些有意思的問(wèn)題,值得memo一下。
先來(lái)看段代碼:
- #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();
- }
大家看得出這段代碼中的問(wèn)題嗎?(呵呵,不要告訴我是cuteqt不能訪問(wèn)哦~)
這段代碼ms十分標(biāo)準(zhǔn),非常符合筆者平時(shí)寫Qt程序書(shū)寫main函數(shù)的習(xí)慣, 孰料想竟然是個(gè)錯(cuò)誤的習(xí)慣,而且問(wèn)題很嚴(yán)重哦。 給個(gè)提示:在程序退出時(shí)會(huì)aborted。
如果還沒(méi)想出來(lái)是什么問(wèn)題,嘿嘿,沒(méi)關(guān)系,看了下面的答案你就明白了。
在這段程序里QApplication實(shí)例創(chuàng)建在stack上,生命期是main的大括號(hào)內(nèi),而mw則通過(guò)new創(chuàng)建在heap上,在程序退出時(shí)才會(huì)被析構(gòu)。 換句話說(shuō),mw的生存期長(zhǎng)于application的生存期…..這可是Qt編程的大忌, 因?yàn)樵赒t中所有的Paint Device都必須要在有QApplication實(shí)例的情況下創(chuàng)建和使用。
不過(guò)如果把這個(gè)程序?qū)懗鰜?lái)運(yùn)行一下,未必會(huì)出現(xiàn)我說(shuō)的aborted的問(wèn)題,大多數(shù)代碼類似的程序都能安全的運(yùn)行(這也是為什么用了那么多年的Qt從來(lái)沒(méi)有注意過(guò)這個(gè)問(wèn)題,并且養(yǎng)成了我錯(cuò)誤的編程習(xí)慣。)。這里的trick在于application退出時(shí)mw已經(jīng)被關(guān)閉,mw中的所有Paint Device一般都不會(huì)被訪問(wèn)到了, 所以這個(gè)錯(cuò)誤隱藏在很深的陰暗角落, 偷偷地嘲笑我們呢!
要想試驗(yàn)這個(gè)問(wèn)題也很簡(jiǎn)單,把load的參數(shù)換成本地文件 test.html, 并把下面的內(nèi)容寫進(jìn)test.html就能看到拉:
- <form>
- <select id="headertest">
- <option>Item1</option>
- <option>Item2</option>
- <option>Item3</option>
- </select>
- </form>
這個(gè)html里使用了下拉選單。如果你運(yùn)行程序并點(diǎn)開(kāi)該選單,之后退出程序你就會(huì)看到Aborted錯(cuò)誤提示,并打印出錯(cuò)誤信息:“QWidget: Must construct a QApplication before a QPaintDevice”。
既然提出的問(wèn)題,當(dāng)然也要給出解決的方案。 有兩種可行的方法避免該錯(cuò)誤。 一個(gè)當(dāng)然是糾正一下編程習(xí)慣,對(duì)mw不要用new的方式創(chuàng)建,改在stack上創(chuàng)建,如下代碼:
- #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解決此問(wèn)題,想辦法讓mw在application之前clean up,那就是用WA_DeleteOnClose屬性。該屬性標(biāo)示窗體會(huì)在close時(shí)被析構(gòu),這樣就保證不會(huì)留存在application析構(gòu)之后了,是個(gè)很好的辦法。
代碼如下:
- #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();
- }
發(fā)現(xiàn)問(wèn)題和解決問(wèn)題是件很有樂(lè)趣的事情,大家不要把時(shí)間都浪費(fèi)在猜測(cè)上,要多動(dòng)手多思考才能進(jìn)步!
Qt Jambi也存在類似的問(wèn)題,如果以程序啟動(dòng)的代碼塊去啟動(dòng)QApplication,在程序運(yùn)行過(guò)程中,一些資源回收會(huì)報(bào)出Null指針錯(cuò)誤,這些錯(cuò)誤,通過(guò)debug,最終都會(huì)指向QWidget這個(gè)類。當(dāng)把QApplication啟動(dòng)的執(zhí)行程序移出main函數(shù),問(wèn)題迎刃而解。要多注意細(xì)節(jié)。
小結(jié):分析Qt編程習(xí)慣 主窗體創(chuàng)建問(wèn)題的內(nèi)容介紹完了,希望本文對(duì)你有所幫助!更多內(nèi)容請(qǐng)參考編輯推薦。