我們都知道,進程就是正在執行的程式。而在Linux中,可以使用一個進程來建立另外一個進程。這樣的話,Linux的進程的組織圖其實有點像Linux分類樹,是個階層的,可以使用pstree命令來查看。在最上面是init程式的執行進程。它是所有進程的老祖宗。Linux提供了兩個函數來建立進程。
1.fork()
fork()提供了建立進程的基本操作,可以說它是Linux系統多任務的基礎。該函數在unistd.h庫中聲明。
#include <stdio.h>#include <unistd.h>#include <stdlib.h>int main(){printf( "建立進程前\n" );pid_t pid = fork();if( !pid ){printf( "我是子進程喲,我的PID是:%d\n" ,getpid() );}else if( pid>0 ){printf( "我是父進程,我的PID是:%d,我的子進程PID是:%d\n",getpid(),pid );}else{printf( "建立進程失敗了喲\n" );exit(1);}return 1;}
在調用fork()之前,只有一個進程,但是fork()之後,將產生一個該進程的子進程,該子進程完全複製父進程,此時父子兩個進程同時運行。在fork()的時候,如果返回的是0,則說明該進程是子進程。如果返回大於0則說明是父進程。如果小於0(其實是-1),則說明建立進程失敗了。
每個進程都有一個唯一標示符,即PID,可以使用getpid()來擷取。父進程返回的pid其實是子進程的pid。
貌似這樣看,fork()之後也沒有什麼作用。其實不然,如果fork()之後跟其他linux功能使用,還是用處很大的。比如我們可以在父子進程中通過通訊協定來通訊,就可以協同完成一些任務了。
2.exec系列函數
如果只有fork(),肯定是不完美的,因為fork()只能參數一個父進程的副本。而exec系列函數則可以協助我們建立一個全新的新進程。
int execl( const char *path, const char *arg, ...);int execlp( const char *file, const char *arg, ...);int execle( const char *path, const char *arg , ..., char* const envp[]);int execv( const char *path, char *const argv[]);int execvp( const char *file, char *const argv[]);
以上函數在unistd.h聲明。
下面我們以execl()函數為例:
#include <stdio.h>#include <unistd.h>int main(){execl("/bin/ls","ls","-l",NULL);printf("如果execl執行失敗,這個就會列印出來了\n");return 1;}
該程式運行到execle()時,載入ls程式,並且覆蓋當前程式的空間。這樣就參數了一個新的進程,但是注意,這個新進程的PID跟載入它的進程是一樣的。
3.fork()和exec()一起調用
fork()可以建立子進程,但是子進程只是父進程的副本。我們可以利用exec()函數在子進程來重新載入一個全新的進程。下面看一個兩個函數聯用的列子。
#include <stdio.h>#include <stdlib.h>#include <unistd.h>int main(){pid_t pid = fork();switch( pid ){case 0:printf("子進程\n");execl("/bin/ls","ls","-l",NULL);case -1:printf("fork失敗了\n");exit(1);default:wait(NULL);printf("完成了喲!\n");exit(0);}}
首先,fork建立子進程,然後在子進程中使用execl()產生一個ls程式的進程。而父進程則調用wait()來等待,直到子進程調用結束。