Posted on 2009-04-15 22:48
Prayer 閱讀(2931)
評(píng)論(1) 編輯 收藏 引用 所屬分類:
LINUX/UNIX/AIX
fork () 的作用是什么?換句話說,你用 fork () 的目的是什么?
――是為了產(chǎn)生一個(gè)新的進(jìn)程,地球人都知道
產(chǎn)生一個(gè)什么樣的進(jìn)程?
――和你本來調(diào)用 fork () 的那個(gè)進(jìn)程基本一樣的進(jìn)程,其實(shí)就是你原來進(jìn)程的副本
真的完全一樣嗎?
――當(dāng)然不能完全一樣,你要兩個(gè)除了 pid 之外其它一模一樣的進(jìn)程干什么,就算 cpu mem
再多也不用這么擺譜吧?
哪里不一樣?
――當(dāng)然最重要的是 fork () 之后執(zhí)行的代碼不一樣,you know, i know
怎么實(shí)現(xiàn)呢?
――如果是 Windows,它會(huì)讓你在 fork () 里面提供一大堆東西,指明這個(gè)那個(gè)什么的……
我用的是 unix 啊
――所以很簡(jiǎn)單,unix 會(huì)讓兩個(gè)進(jìn)程(不錯(cuò),原來是一個(gè),unix 替你復(fù)制了一個(gè),現(xiàn)在有兩個(gè)):
在 fork () 之后產(chǎn)生不同:返回值不同。其中一個(gè)進(jìn)程(使用新的 pid)里面的 fork () 返回零,
這個(gè)進(jìn)程就是“子進(jìn)程”;而另一個(gè)進(jìn)程(使用原來的 pid)中的 fork () 返回前面那個(gè)子進(jìn)程的
pid,他自己被稱為“父進(jìn)程”
然后呢?
――寫代碼的人又不笨,當(dāng)然就根據(jù)返回值是否非零來判斷了,現(xiàn)在我是在子進(jìn)程里面呢,還是在
父進(jìn)程里面?在子進(jìn)程里面就執(zhí)行子進(jìn)程該執(zhí)行的代碼,在父進(jìn)程里面就執(zhí)行父進(jìn)程的代碼……
有鐵桿 windows fans 借此說明,windows 好啊,子進(jìn)程用子進(jìn)程的代碼,父進(jìn)程用父進(jìn)程的,
你 unix 笨了吧,子進(jìn)程包含父進(jìn)程、子進(jìn)程的代碼,父進(jìn)程包含父進(jìn)程子進(jìn)程的代碼,豈不是多"
占用內(nèi)存了嗎?
共享同一代碼段,增加的只是全局共享數(shù)據(jù)和對(duì)文件描述符的引用等,另外就是堆棧。你一個(gè)代碼:
長(zhǎng)達(dá) 10M 的進(jìn)程,fork () 出三四個(gè)子進(jìn)程,只是增加一點(diǎn)內(nèi)存占用(如果你沒有使用很多全局變量
的話),而不是占用 40M 以上的內(nèi)存。
父進(jìn)程里面有個(gè)變量 var,子進(jìn)程里面也有個(gè)變量 var
linux 是多用戶和多進(jìn)程的操作系統(tǒng),進(jìn)程在操作系統(tǒng)中的創(chuàng)建,都會(huì)生成一個(gè)進(jìn)程描述塊,描述當(dāng)前進(jìn)程的所有信息,包括,數(shù)據(jù)段、代碼段、堆棧段的地址,當(dāng)前進(jìn)程的環(huán)境變量,文件描述符等。
fork函數(shù)過程:操作系統(tǒng)先創(chuàng)建一個(gè)進(jìn)程描述塊,然后把父進(jìn)程的所有進(jìn)程描述符的信息精確拷貝過來,和父進(jìn)程一樣(除了進(jìn)程ID不一樣外),代碼段共享,數(shù)據(jù)段和堆棧段復(fù)制,所有的寄存器的值全部精確拷貝,文件描述符也許精確拷貝。
fork的返回值,fork在父進(jìn)程空間中返回子進(jìn)程的PID,在子進(jìn)程空間中返回0。