linux中fork和exec
學(xué)過C語言的都知道,Unix下某個(gè)進(jìn)程的內(nèi)存分成三部分:代碼段,堆棧段,數(shù)據(jù)段。代碼段用來存放程序運(yùn)行的代碼,堆棧段用來存放子程序的局部變量,數(shù)據(jù)段用來存放全局變量。這在perl里也是一樣的。
perl的fork調(diào)用,跟C的一樣,當(dāng)發(fā)生fork調(diào)用時(shí),實(shí)際上發(fā)生如下事:
父進(jìn)程將代碼段,堆棧段,數(shù)據(jù)段完全復(fù)制一份給子進(jìn)程。也就是說,在子進(jìn)程運(yùn)行之初,它擁有父進(jìn)程的一切變量和句柄。例如,父進(jìn)程申明了某個(gè)hash表,那這個(gè)hash表也會(huì)被子進(jìn)程擁有。
然而,一旦子進(jìn)程開始運(yùn)行,它的數(shù)據(jù)段和堆棧段就在內(nèi)存里完全和父進(jìn)程分離開了。也就是說,兩個(gè)進(jìn)程間不再共享任何數(shù)據(jù)。例如前面所說的hash表,雖然子進(jìn)程從父進(jìn)程處繼承了這個(gè)數(shù)據(jù)結(jié)構(gòu),但子進(jìn)程寫往hash里的數(shù)據(jù),不會(huì)被父進(jìn)程訪問到。在shell里用ps命令,可以看到2個(gè)獨(dú)立運(yùn)行的進(jìn)程。通常你 kill掉1個(gè),不會(huì)影響另1個(gè)的運(yùn)行。
那么父進(jìn)程和fork出來的子進(jìn)程如何通信呢?父進(jìn)程和子進(jìn)程間的通信有多種方法,最常見的是信號(hào),另外還有管道,Socket,消息隊(duì)列等,不在這里詳敘。而2個(gè)進(jìn)程間共享數(shù)據(jù)的辦法,可以用線程或共享內(nèi)存,我對(duì)這方面不熟悉。
如果大概明白了fork,那么exec就容易理解了。一個(gè)進(jìn)程一旦調(diào)用exec類函數(shù),它本身就“死亡”了,系統(tǒng)把代碼段替換成新的程序的代碼,廢棄原有的數(shù)據(jù)段和堆棧段,并為新程序分配新的數(shù)據(jù)段與堆棧段,唯一留下的,就是進(jìn)程號(hào),也就是說,對(duì)系統(tǒng)而言,還是同一個(gè)進(jìn)程,不過已經(jīng)是另一個(gè)程序了。
在perl里,調(diào)用exec后,原進(jìn)程就完全消失,由于消失了,它也就不會(huì)從新進(jìn)程接受到任何返回值,除非新進(jìn)程意外終止,原進(jìn)程會(huì)接受到錯(cuò)誤值。
總結(jié)
以上是生活随笔為你收集整理的linux中fork和exec的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: C++的类和C里面的struct有什么区
- 下一篇: 关于Javaweb部署到linux服务器