exec函式族
fork()函式創建子行程后,子行程往往要呼叫一種e x e c函式以執行另一個程式,當行程呼叫一種exec函式時,該行程完全由新程式代換,而新程式則從其 ma i n函式開始執行, 因為呼叫exec并不創建新行程,所以前后的行程ID并未改變,exec只是用另一個新程式替換了當前行程的正文、資料、堆和堆疊段, 有六種不同的e x e c函式可供使用,它們常常被統稱為exec函式,
#include<unistd.h> int execl(const char *pathname, const char * arg0,.../*(char*)0*/); int execv(const char *pathname, char *const argv[]); int execle(const char *pathname,const char * arg0,.../*(char*)0,char *const envp[]*/); int execve(const char *pathname, char *const argv[],char * const envp[]); int execlp(const char *filename, const char * arg0,.../*(char*)0*/); int execvp(const char *filename, char *const argv[]); //六個函式回傳:若出錯則為-1,若成功則不回傳
這些函式之間的第一個區別是前四個取路徑名作為引數,后兩個則取檔案名作為引數,當指定filename作為引數時:
如果filename中包含/,則就將其視為路徑名,否則就按PAT H環境變數,在有關目錄中搜尋可執行檔案,
PATH變數包含了一張目錄表 (稱為路徑前綴),目錄之間用冒號( : )分隔,例如下列na me = value環境字串:
PATH = /bin:/usr/bin:/usr/local/bin: 指定在四個目錄中進行搜索,(零長前綴也表示當前目錄,在value的開始處可用:表示,在行中間則要用::表示,在行尾以:表示,)
如果execlp和execvp中的任意一個使用路徑前綴中的一個找到了一個可執行檔案,但是該檔案不是由連接編輯程式產生的機器可執行代碼檔案,則就認為該檔案是一個 shell腳本,于是 試著呼叫/bin/sh,并以該filename作為shell的輸入,第二個區別與引數表的傳遞有關 ( l表示表( l i s t ),v表示矢量( v e c t o r ) ),函式execl、execlp和execle要求將新程式的每個命令列引數都說明為一個單獨的引數,這種引數表以空指標結尾,
對于另外三個函式( execv, execvp和execve ),則應先構造一個指向各引數的指標陣列,然后將該陣列地址作為這三個函式的引數, 在使用ANSIC原型之前,對execl , execle和execlp三個函式表示命令列引數的一般方法是: char *arg 0, char *arg 1, ..., char *arg n, (char *) 0 應當特別指出的是:在最后一個命令列引數之后跟了一個空指標,如果用常數 0來表示一個空指標,則必須將它強制轉換為一個字符指標,否則它將被解釋為整型引數,如果一個整型數的長度與char *的長度不同,exec函式實際引數就將出錯, 最后一個區別與向新程式傳遞環境表相關,以e結尾的兩個函式( execle和execve)可以傳遞一個指向環境字串指標陣列的指標,其他四個函式則使用呼叫行程中的 environ變數為新程式復制現存的環境,通常,一個行程允許將其環境傳播給其子行程,但有時也有這種情況,行程想要為子行程指定一個確定的環境,
例如,在初始化一個新登錄的 shell時,login程式創建一個只定義少數幾個變數的特殊環境,而在我們登錄時,可以通過 shell啟動檔案,將其他變數加到環境中,
在使用 ANSI C原型之前, execle 的引數是:char * pathname, char *arg 0, ?, char *a rg n, (char *)0, char *envp[ ] 從中可見,最后一個引數是指向環境字串的各字符指標構成的陣列的指標,而在 ANSIC原型中,所有命令列引數,包括空指標,e n v p指標都用省略號(?)表示, 這六個e x e c函式的引數很難記憶,函式名中的字符會給我們一些幫助,字母 p表示該函式
取filename作為引數,并且用PATH環境變數尋找可執行檔案,字母l表示該函式取一個引數表,它與字母v互斥,v表示該函式取一個arg v[ ],最后,字母e表示該函式取e nvp[ ]陣列,而不使用當前環境
這六個函式中只有一個execve是內核的系統呼叫,另外五個只是庫函式,它們最終都要呼叫系統呼叫,這六個函式之間的關系示于圖8 - 2中,在這種安排中,庫函式 execlp 和execvp 使用PATH環境變數查找第一個包含名為filename的可執行檔案的路徑名前綴,


例1:
1 #include<stdio.h> 2 #include<stdlib.h> 3 #include<unistd.h> 4 5 int main(int argc,char * argv[]) 6 { 7 if(argc<2) 8 { 9 perror("you haven't input the filename,please try again!\n"); 10 exit(EXIT_FAILURE); 11 } 12 if(execl("./create_file","create_file",argv[1],NULL)<0) 13 perror("execl error"); 14 }
可以看見的是有一個名為create_file的可執行檔案用于創建檔案,在這里我們使用execl函式替換程式,


例2:
代碼:
1 #include<stdio.h> 2 #include<stdlib.h> 3 #include<unistd.h> 4 5 int main(int argc,char * argv[]) 6 { 7 if(argc<2) 8 { 9 perror("you haven't input the filename,please try again!\n"); 10 exit(EXIT_FAILURE); 11 } 12 if(execl("/bin/touch","touch",argv[1],NULL)<0) 13 perror("execl error"); 14 }
可以看見我們將exec2 成功的替換為了touch,并成功的執行了,

參考資料
Linux/Unix系統編程手冊
Unix環境高級編程
Linux程式設計
轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/82008.html
標籤:Linux
上一篇:Linux 命令列下搜索工具大盤點,效率提高不止一倍!
下一篇:運維相關
