深入分析父子線程、進(jìn)程終止順序不同產(chǎn)生的結(jié)果
Linux下編程,線程、進(jìn)程退出順序問(wèn)題紛紛擾擾,如果父進(jìn)程/線程先于子進(jìn)程/線程終止,系統(tǒng)會(huì)做什么處理呢?反之,如果子進(jìn)程/線程先于父進(jìn)程/線程終止,系統(tǒng)又會(huì)做什么處理呢?下面是我個(gè)人平時(shí)在筆記上做的一些象征性的總結(jié),如果有疑問(wèn)之處,大可提出,我一直認(rèn)為懷疑是人類(lèi)進(jìn)步的根本所在。
一、線程
Linux線程創(chuàng)建函數(shù)為pthread_create(),默認(rèn)規(guī)則是誰(shuí)創(chuàng)建子線程,誰(shuí)就要負(fù)責(zé)子線程的資源回收,當(dāng)父線程退出后,子線程也隨著退出。所以,一般情況下,父線程退出時(shí)都要確保子線程已經(jīng)退出,所以會(huì)使用pthread_join()函數(shù)阻塞等待子線程的退出信號(hào)/標(biāo)識(shí)。
pthread_detach(threadid)函數(shù)的功能是使線程ID為threadid的線程處于分離狀態(tài)(可以為非父子關(guān)系),一旦線程處于分離狀態(tài),該線程終止時(shí)底層資源立即被回收;否則終止子線程的狀態(tài)會(huì)一直保存占用系統(tǒng)的資源直到主線程調(diào)用pthread_join(threadid,NULL)獲取線程的退出狀態(tài)。被創(chuàng)建的子線程也可以自己分離自己,子線程調(diào)用pthread_detach(pthread_self())就是分離自己,因?yàn)閜thread_self()這個(gè)函數(shù)返回的就是自己本身的線程ID。
1)父線程先于子線程終止
父線程先于子線程,則子線程為異常退出 ,那肯定沒(méi)有使用阻塞非分離函數(shù)pthread_join,分2種情況:
a)子線程已與父線程分離,如調(diào)用線程分離函數(shù)pthread_detach,則資源被自動(dòng)回收釋放。
b)子線程未與父線程分離,則資源無(wú)法釋放,造成了資源浪費(fèi)和系統(tǒng)臃腫(這種情況,我看有些資料上說(shuō)系統(tǒng)也能自動(dòng)釋放子線程的資源,如關(guān)閉描述符,釋放內(nèi)存空間等等,但個(gè)人做過(guò)一些測(cè)試,比如在子線程中分配很多空間等,進(jìn)程退出后,top查看內(nèi)存狀態(tài)時(shí)還存在)。
2)子線程先于父線程終止
也分2種情況:
a)正常情況:子線程調(diào)用了線程分離函數(shù)ptread_detach(),或父線程調(diào)用了等待線程結(jié)束函數(shù)pthread_join()。
b)異常情況:如果上面二者都為調(diào)用,則為子線程分配的資源無(wú)法得到釋放。
二、進(jìn)程
一個(gè)現(xiàn)有進(jìn)程可以調(diào)用fork函數(shù)創(chuàng)建一個(gè)新進(jìn)程。由fork創(chuàng)建的新進(jìn)程被稱為子進(jìn)程(child process)。fork函數(shù)被調(diào)用一次但返回兩次。兩次返回的唯一區(qū)別是子進(jìn)程中返回0值而父進(jìn)程中返回子進(jìn)程ID。
1)父進(jìn)程先于子進(jìn)程終止
當(dāng)父進(jìn)程先退出時(shí),系統(tǒng)會(huì)讓init進(jìn)程接管子進(jìn)程,該子線程成為了孤兒進(jìn)程。
2)子進(jìn)程先于父進(jìn)程終止
分為2種情況:
a)正常情況:父進(jìn)程調(diào)用了wait函數(shù) (非父子進(jìn)程則用waitpid函數(shù)),此時(shí)父進(jìn)程會(huì)等待子進(jìn)程結(jié)束。
b)父進(jìn)程又沒(méi)有調(diào)用wait函數(shù) (非父子進(jìn)程則未調(diào)用waitpid函數(shù)),此種情況子進(jìn)程進(jìn)入僵死狀態(tài)即僵尸進(jìn)程,并且會(huì)一直保持下去直到系統(tǒng)重啟。子進(jìn)程處于僵死狀態(tài)時(shí),內(nèi)核只保存進(jìn)程的一些必要信息以備父進(jìn)程所需。此時(shí)子進(jìn)程始終占有著資源,同時(shí)也減少了系統(tǒng)可以創(chuàng)建的最大進(jìn)程數(shù)。
僵死狀態(tài):一個(gè)已經(jīng)終止、但是其父進(jìn)程尚未對(duì)其進(jìn)行善后處理(獲取終止子進(jìn)程的有關(guān)信息,釋放它仍占有的資源)的進(jìn)程被稱為僵死進(jìn)程(zombie)。ps命令將僵死進(jìn)程的狀態(tài)打印為Z 。
相關(guān)文章
VC++ loadlibrary()加載三方dll失敗, 返回錯(cuò)誤碼:126的解決方法
今天在編寫(xiě)VC++ loadlibrary()加載三方dll是總是失敗,并且返回錯(cuò)誤碼:126,這里就為大家分享一下具體的解決方法2021-03-03
C++回調(diào)函數(shù)實(shí)現(xiàn)計(jì)算器和qsort
這篇文章主要介紹了C++回調(diào)函數(shù)實(shí)現(xiàn)計(jì)算器和qsort,回調(diào)函數(shù)就是一個(gè)通過(guò)函數(shù)指針調(diào)用的函數(shù)。如果你把函數(shù)的指針(地址)作為參數(shù)傳遞給另一個(gè)函數(shù),當(dāng)這個(gè)指針被用來(lái)調(diào)用其所指向的函數(shù)時(shí),我們就說(shuō)這是回調(diào)函數(shù)2022-08-08
C++開(kāi)發(fā)之CRC校驗(yàn)實(shí)例詳解
這篇文章主要介紹了C++開(kāi)發(fā)之CRC校驗(yàn)實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-07-07
一文詳解C語(yǔ)言中文件相關(guān)函數(shù)的使用
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言中文件相關(guān)函數(shù)的使用,可以實(shí)現(xiàn)文件的讀寫(xiě)、打開(kāi)和關(guān)閉。文中通過(guò)示例進(jìn)行了詳細(xì)介紹,需要的可以參考一下2022-07-07
C++中實(shí)現(xiàn)把表的數(shù)據(jù)導(dǎo)出到EXCEL并打印實(shí)例代碼
這篇文章主要介紹了實(shí)現(xiàn)把表的數(shù)據(jù)導(dǎo)出到EXCEL并打印實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-04-04
C語(yǔ)言?struct結(jié)構(gòu)體超詳細(xì)講解
C語(yǔ)言中,結(jié)構(gòu)體類(lèi)型屬于一種構(gòu)造類(lèi)型(其他的構(gòu)造類(lèi)型還有:數(shù)組類(lèi)型,聯(lián)合類(lèi)型),下面這篇文章主要給大家介紹了關(guān)于C語(yǔ)言結(jié)構(gòu)體(struct)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-04-04
C++?shared_ptr智能指針reset()使用示例詳解
這篇文章主要為大家介紹了C++?shared_ptr智能指針reset()使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08
C++通過(guò)boost.date_time進(jìn)行時(shí)間運(yùn)算
這篇文章介紹了C++通過(guò)boost.date_time進(jìn)行時(shí)間運(yùn)算的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06

