色婷婷AⅤ一区二区三区|亚洲精品第一国产综合亚AV|久久精品官方网视频|日本28视频香蕉

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > 我的QT5學習之路(二)――第一個程序

          我的QT5學習之路(二)――第一個程序

          作者: 時間:2018-08-02 來源:網(wǎng)絡(luò) 收藏

          一、前言  “工欲善其事,必先利其器”,上一節(jié),我介紹了Qt的安裝和配置方法,搭建了基本的開發(fā)平臺。這一節(jié),來通過一個簡單的例子來了解Qt的編程樣式和規(guī)范,開始嘍~~~

          本文引用地址:http://cafeforensic.com/article/201808/385046.htm

          回到頂部(go to top)

          二、第一個程序——Hello World  首先,我們可以按照上一節(jié)的方法建立一個新的工程,工程的名字可以就叫做Hello,隨你的便。在創(chuàng)建工程的過程中,有一個選擇是否創(chuàng)建視圖界面的選項,這個可以先不選擇,因為我們現(xiàn)在只是了解Qt的機制,不需要Qt幫我們做太多的事情,創(chuàng)建完成后,打開main.cpp。

          我做的工作主要就是:

          1、屏蔽掉程序自己的對話框程序代碼;

          2、添加一個label控件,并給他傳一個文本值,最后顯示。

          最后的顯示結(jié)果:

          其次,我們來分析一下Qt的基本流程。1~3行是頭文件包含,這里有兩種頭文件,第一種是自定義頭文件或者本地頭文件,用“ ”來進行表示和包含;第二種是系統(tǒng)頭文件,這里就是Qt自帶的頭文件,直接用>進行表示和包含就可以。第7行是創(chuàng)建一個QApplication的實例,對于 Qt 程序來說,main()函數(shù)一般以創(chuàng)建 application 對象(GUI 程序是QApplication,非 GUI 程序是QCoreApplication。QApplication實際上是QCoreApplication的子類。),這個對象用于管理 Qt 程序的生命周期,開啟事件循環(huán)。10~11行是核心代碼,也就是我們實際添加的用例代碼,這里我創(chuàng)建了一個QLabel,利用構(gòu)造函數(shù)對其進行賦值操作,最后調(diào)用show方法將其顯示出來。最后一行調(diào)用exec,開啟事件循環(huán)(可以理解成一段無線循環(huán))。

          寫完這兩句代碼之后,我們想一個問題,這里我們先不討論Qt的消息機制和其他的通信原理,單純從C++方面考慮程序的穩(wěn)定性和魯棒性。

          問題1:我創(chuàng)建的QLabel是創(chuàng)建在棧上的還是堆上的?

          問題2:如果我把QLabel變量創(chuàng)建為堆上變量,應(yīng)該注意哪些問題?

          我們先來討論問題1,這個應(yīng)該沒有什么爭議,Qlabel變量是創(chuàng)建在棧上的。再來看看問題2,如果我因為某些需求將變量聲明為堆上變量,那么這個時候我就要給這個變量分配空間。這個時候問題就來了,分配空間了,程序結(jié)束后誰來釋放啊?內(nèi)存泄露了怎么辦啊?怎么能夠防止內(nèi)存泄露啊?如果我們對其不管不顧,在程序結(jié)束后,操作系統(tǒng)會將其回收,但是,我們看到label 是建立在堆上的,app 是建立在棧上的。這意味著,label 會在 app 之后析構(gòu)。也就是說,label 的生命周期長于 app 的生命周期。這可是 Qt 編程的大忌。因為在 Qt 中,所有的QPaintDevice必須要在有QApplication實例的情況下創(chuàng)建和使用。大家好奇的話,可以提一句,QLabel繼承自QWidget,QWidget則是QPaintDevice的子類。之所以上面的代碼不會有問題,是因為 app 退出時,label 已經(jīng)關(guān)閉,這樣的話,label 的所有QPaintDevice一般都不會被訪問到了。但是,如果我們的程序,在 app 退出時,組件卻沒有關(guān)閉,這就會造成程序崩潰。

          此外,這里的程序沒有崩潰的另一個原因是如果在主函數(shù)結(jié)尾,可以不釋放;在其它區(qū)域結(jié)尾,new出來的內(nèi)存是逆序釋放的,這是c++標準的規(guī)定。

          這個時候,或許知道C++11標準的童鞋想到了智能指針。沒錯,智能指針是可以作為指針的托管類來實現(xiàn)指針的自動釋放,但是智能指針如果用不好同樣會產(chǎn)生各種各樣的問題,因此,建議剛開始學習的同學,能不用堆上變量就先不要用,如果真要用的話,記得想好內(nèi)容溢出和泄露的問題并采取必要的預(yù)防辦法或者不使用智能指針,給變量添加屬性。

          label->setAttribute(Qt::WA_DeleteOnClose);

          這時,我們回頭去看exec方法,因為如此,我們在棧上構(gòu)建了QLabel對象,卻能夠一直顯示在那里(試想,如果不是無限循環(huán),main()函數(shù)立刻會退出,QLabel對象當然也就直接析構(gòu)了)。

          最后,為大家附上堆上變量和智能指針的聲明方式,僅供參考。

          2.1 堆上代碼參考


          1 #include

          2 #include

          3

          4 int main(int argc ,char **argv)

          5 {

          6 QApplication a(argc,argv);

          7 QLabel *label =new QLabel(Hello world);

          8 label->show();

          9

          10 return a.exec();

          11 }


          2.2 智能指針代碼參考(Sailfish OS)


          1 #include

          2

          3 int main(int argc, char *argv[])

          4 {

          5 QScopedPointer app(new QApplication(argc, argv));

          6 QScopedPointer view(new QQuickView);

          7 view->setSource(/path/to/main.qml);

          8 ...

          9 return app->exec();

          10 }


          三、我的觀點  關(guān)于指針的使用方式和地方的選擇這個確實是見仁見智的,我自己對于指針的使用是很小心的,如果使用的話也會在一些不牽扯線程安全的情況下使用,并且打印日志報告。另外一個問題就是指針的釋放和重用問題,我的觀點是如果在指針被釋放的作用域進行delete的操作,但是并沒有置為null,這個時候指針應(yīng)該是還能夠使用的,只是沒有交還給操作系統(tǒng)而已,如有誤解,請指正。

          no pains ,no gains. 給自己加油,為未來奮斗。



          關(guān)鍵詞: QT5

          評論


          技術(shù)專區(qū)

          關(guān)閉