詳解linux中fork、vfork、clone函數(shù)的區(qū)別
在linux系統(tǒng)中,fork(),vfork()和clone函數(shù)都可以創(chuàng)建一個進程,但是它們的區(qū)別是什么呢???本文就這三者做一個較深入的分析?。?!
1.fork()
fork()函數(shù)的作用是創(chuàng)建一個新進程,由fork創(chuàng)建的進程稱為子進程,fork函數(shù)調(diào)用一次返回兩次,子進程返回值為0,父進程返回子進程的進程ID。我們知道,一個進程的地
址空間主要由代碼段,數(shù)據(jù)段,堆和棧構(gòu)成,那么p2就要復(fù)制相關(guān)的段到物理內(nèi)存。原始的unix系統(tǒng)的實現(xiàn)的是一種傻
瓜式的進程創(chuàng)建,這些復(fù)制包括:
(1) 為子進程的頁表分配頁面,確定頁表的位置;
(2)為子進程的頁分配頁面,確定子進程頁面的位置;
(3)初始化子進程的頁表;
(4)把父進程的頁復(fù)制到子進程對應(yīng)的頁中
從圖中我們可以看出除了正文段外,子進程的所有其它段都分配了物理空間,并將父進程的相關(guān)內(nèi)容拷貝過來。父進程的task_struct結(jié)構(gòu)中的打開文件描述符,進程組ID,
回話ID都進行復(fù)制。
下面通過簡單的代碼檢測一下fork()函數(shù):
#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys types.h=""> #include <pthread.h> #include #include <iostream> using namespace std; int main() { int num = 1; int child; if(!(child =fork())) { cout<<&num<<endl; address:="" alt="\" child="vfork()))" class="brush:java;" else="" father="" img="" include="" int="" is:="" namespace="" num="1;" num:="" pid="" pre="" son="" src="/uploadfile/Collfiles/20170421/201704210949041534.jpg" sys="" types.h="" using=""><p>測試結(jié)果:</p><p><img alt="\" src="/uploadfile/Collfiles/20170421/201704210949041535.jpg" style="width: 630px; height: 201.393px;"></p><p>從測試結(jié)果中我們可以看到,在子進程修改了num變量的值后,父進程的num的值也發(fā)生改變,說明對于子進程和父進程來說,它們操作的是同一個地方的num值,下面就是vfork的示意圖:</p><p><img alt="\" src="/uploadfile/Collfiles/20170421/201704210949041536.jpg" style="width: 508px; height: 375px;"></p><p>可以看出子進程直接共享了父進程的虛擬進程空間。</p><p>3.clone()</p><p> clone()函數(shù)是linux系統(tǒng)中,用來創(chuàng)建輕量級進程。</p><p>函數(shù)原形:</p><pre class="brush:java;">int clone(int (*fn)(void *), void *child_stack, int flags, void *arg); 下面是flags可以取的值 標志 含義 CLONE_PARENT 創(chuàng)建的子進程的父進程是調(diào)用者的父進程,新進程與創(chuàng)建它的進程成了“兄弟”而不是“父子” CLONE_FS 子進程與父進程共享相同的文件系統(tǒng),包括root、當前目錄、umask CLONE_FILES 子進程與父進程共享相同的文件描述符(file descriptor)表 CLONE_NEWNS 在新的namespace啟動子進程,namespace描述了進程的文件hierarchy CLONE_SIGHAND 子進程與父進程共享相同的信號處理(signal handler)表 CLONE_PTRACE 若父進程被trace,子進程也被trace CLONE_VFORK 父進程被掛起,直至子進程釋放虛擬內(nèi)存資源 CLONE_VM 子進程與父進程運行于相同的內(nèi)存空間 CLONE_PID 子進程在創(chuàng)建時PID與父進程一致 CLONE_THREAD Linux 2.4中增加以支持POSIX線程標準,子進程與父進程共享相同的線程群 下面的例子是創(chuàng)建一個線程(子進程共享了父進程虛存空間,沒有自己獨立的虛存空間不能稱其為進程)。父進程被掛起當子線程釋放虛 存資源后再繼續(xù)執(zhí)行。</pre> 測試代碼1: <pre class="brush:java;">#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys types.h=""> #include <pthread.h> #include #include <iostream> using namespace std; #define FIBER_STACK 8192 int a; void * stack; int func(void *){ cout<<&a<<endl; a="" char="" creating="" int="" is="" is:="" my="" pid="" pre="" son="" stack="" the="" this="" void=""> 結(jié)果:<p><img alt="\" src="/uploadfile/Collfiles/20170421/201704210949041537.png" style="width: 630px; height: 201.393px;"></p><p>測試代碼2(做如下修改):</p><pre class="brush:java;">clone(func, (char *)stack + FIBER_STACK,CLONE_VFORK, 0);</pre> 結(jié)果: <p><img alt="\" src="http://www.dbjr.com.cn/uploadfile/Collfiles/20170421/201704210949041538.png" style="display: block; width: 630px; height: 201.393px;"></p> <p>很明顯,在測試2中將CLONE_VM刪掉之后,子進程和父進程就不會公用頁表,子進程創(chuàng)建新的頁表。從某種意義上來說,clone其實是fork和vfrok的更高層次版本,,它們的關(guān)</p> <p>系如下(《深入理解linux內(nèi)核》中描述):</p> <p> <strong>傳統(tǒng)的fork()系統(tǒng)調(diào)用在Linux中是用clone()實現(xiàn)的,其中clone()的flags參數(shù)指定為sigchld信號以及所有清0的clone標志,而它的child_stack參數(shù)是父進程當前的堆棧</strong></p> <p><strong>指針,因此,父進程和子進程暫時共享一個用戶態(tài)堆棧。而vfork函數(shù)系統(tǒng)調(diào)用也是用clone實現(xiàn)的,其中clone()的參數(shù)flags指定為sigchld和CLONE_VFORK和CLONE_VM標</strong></p> <p><strong>志,clone()的參數(shù)child_stack等于父進程當前的棧指針?。?!</strong></p> 。只是有一點不明白,把int a和void * stack挪到main函數(shù)里面之后,就會出現(xiàn)編譯錯誤,顯示未定義a和stack,這點有些不懂,望高人指點?。。。?</endl;></iostream></assert.h></pthread.h></sys></unistd.h></stdlib.h></stdio.h></pre> </endl;></iostream></assert.h></pthread.h></sys></unistd.h></stdlib.h></stdio.h>
以上所述是小編給大家介紹的linux中fork、vfork、clone函數(shù)的區(qū)別,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!