Qt的內(nèi)存管理
剛學(xué)Qt時(shí),發(fā)現(xiàn)代碼里到處都是new,而基本上見(jiàn)不到delete,百度得知,Qt中內(nèi)存管理機(jī)制:當(dāng)new一個(gè)對(duì)象指定父親時(shí),父親被銷(xiāo)毀時(shí),會(huì)一并把自對(duì)象銷(xiāo)毀。
在QT的程序中經(jīng)常會(huì)看到只有new而不delete的情況,其實(shí)是因?yàn)镼T有一套回收內(nèi)存的機(jī)制,主要的規(guī)則如下:
1.所有繼承自QOBJECT類(lèi)的類(lèi),如果在new的時(shí)候指定了父親,那么它的清理時(shí)在父親被delete的時(shí)候delete的,所以如果一個(gè)程序中,所有的QOBJECT類(lèi)都指定了父親,那么他們是會(huì)一級(jí)級(jí)的在最上面的父親清理時(shí)被清理,而不用自己清理;
2.程序通常最上層會(huì)有一個(gè)根的QOBJECT,就是放在setCentralWidget()中的那個(gè)QOBJECT,這個(gè)QOBJECT在new的時(shí)候不必指定它的父親,因?yàn)檫@個(gè)語(yǔ)句將設(shè)定它的父親為總的QAPPLICATION,當(dāng)整個(gè)QAPPLICATION沒(méi)有時(shí)它就自動(dòng)清理,所以也無(wú)需清理。9這里QT4和QT3有不同,QT3中用的是setmainwidget函數(shù),但是這個(gè)函數(shù)不作為里面QOBJECT的父親,所以QT3中這個(gè)頂層的QOBJECT要自行銷(xiāo)毀)。
3.這是有人可能會(huì)問(wèn)那如果我自行delete掉這些QT接管負(fù)責(zé)銷(xiāo)毀的指針了會(huì)出現(xiàn)什么情況呢,如果時(shí)這樣的話,正常情況下QT的擁有這個(gè)對(duì)象的那個(gè)父親會(huì)知道這件事情,它會(huì)直到它的兒子被你直接DELETE了,這樣它會(huì)將這個(gè)兒子移出它的列表,并且重新構(gòu)建顯示內(nèi)容,但是直接這樣做時(shí)有風(fēng)險(xiǎn)的!也就是要說(shuō)的下一條
4.當(dāng)一個(gè)QOBJECT正在接受事件隊(duì)列時(shí)如果中途被你DELETE掉了,就是出現(xiàn)問(wèn)題了,所以QT中建議大家不要直接DELETE掉一個(gè)QOBJECT,如果一定要這樣做,要使用QOBJECT的deleteLater()函數(shù),它會(huì)讓所有事件都發(fā)送完一切處理好后馬上清除這片內(nèi)存,而且就算調(diào)用多次的deletelater也不會(huì)有問(wèn)題。
5.QT不建議在一個(gè)QOBJECT 的父親的范圍之外持有對(duì)這個(gè)QOBJECT的指針,因?yàn)槿绻@樣外面的指針很可能不會(huì)察覺(jué)這個(gè)QOBJECT被釋放,會(huì)出現(xiàn)錯(cuò)誤,如果一定要這樣,就要記住你在哪這樣做了,然后抓住那個(gè)被你違規(guī)使用的QOBJECT的destroyed()信號(hào),當(dāng)它沒(méi)有時(shí)趕快置零你的外部指針。當(dāng)然我認(rèn)為這樣做是及其麻煩也不符合高效率編程規(guī)范的,所以如果要這樣在外部持有QOBJECT的指針,建議使用引用或者用智能指針,如QT就提供了智能指針針對(duì)這些情況,見(jiàn)最后一條。
6.QT中的智能指針?lè)庋b為QPointer類(lèi),所有QOBJECT的子類(lèi)都可以用這個(gè)智能指針來(lái)包裝,很多用法與普通指針一樣,可以詳見(jiàn)QT assistant
轉(zhuǎn)自:http://blog.csdn.net/leonwei/article/details/3703598