linux編程 | exec族函數

fork創建子進程後執行的是和父進程相同的程序(但有可能執行不同的代碼分支),子進程往往要調用一種exec函數以執行另一個程序。當進程調用一種exec函數時,該進程的用戶空間代碼和數據完全被新程序替換,從新程序的啓動例程開始執行。調用exec並不創建新進程,所以調用exec前後該進程的id並未改變。

將當前進程的.text、.data替換爲所要加載的程序的.text、.data,然後讓進程從新的.text第一條指令開始執行,但進程ID不變,換核不換殼。

所謂exec函數族,其實有六種以exec開頭的函數,統稱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[]);

int execve(const char *path, char *const argv[], char *const envp[]);

 

exec函數族一般規律

exec函數一旦調用成功即執行新的程序,不返回。只有失敗才返回,錯誤值-1。所以通常我們直接在exec函數調用後直接調用perror()和exit(),無需if判斷。

exec函數族名字很相近,使用起來也很相近,它們的一般規律如下:

l (list)                           命令行參數列表

p (path)                       搜素file時使用path變量

v (vector)                    使用命令行參數數組

e (environment)       使用環境變量數組,不使用進程原有的環境變量,設置新加載程序運行的環境變量

 

帶l的exec函數

這類函數有:execl,execlp,execle

具體說明:表示後邊的參數以可變參數的形式給出且都以一個空指針結束。這裏特別要說明的是,程序名也是參數,所以第一個參數就是程序名。

 

#include <stdio.h>
#include <unistd.h>
 
int main()
{
    printf("process begin...\n");
    execl("/bin/ls", "ls", NULL);
    printf("process end...\n");
    return 0;
}


 

 

帶p的exec函數

這類函數有:execlp,execvp

具體說明:表示第一個參數無需給出具體的路徑,只需給出函數名即可,系統會在PATH環境變量中尋找所對應的程序,如果沒找到的話返回-1。

#include <stdio.h>
#include <unistd.h>
 
int main()
{
    printf("process begin...\n");
    execlp("ls", "ls", "-l", NULL);
    printf("process end...\n");
    return 0;
}
 

 

帶v的exec函數

這類函數有:execv,execvp

具體說明:表示命令所需的參數以char *arg[]形式給出且arg最後一個元素必須是NULL

#include <stdio.h>
#include <unistd.h>
 
int main()
{
    printf("process begin...\n");
    char *argv[] = {"ls", "-a", "-l", NULL};
    execvp("ls", argv);
    printf("process end...\n");
    return 0;
}


帶e的exec函數

這類函數有:execle

具體說明:將環境變量傳遞給需要替換的進程,原來的環境變量不再起作用。

//execle程序
 
#include <stdio.h>
#include <unistd.h>
 
int main()
{
    printf("process begin...\n");
    char *envp[] = {"AA=11", "BB=22", NULL};
    execle("./env", "env", NULL, envp);
    printf("process end...\n");
    return 0;
}
//env程序
 
#include <stdio.h>
 
/* environ本質是{"HOME=/home/alvin", "SHELL=/bin/bash", ...} */
extern char **environ;
 
int main()
{
    int i = 0;
 
    for (i = 0; environ[i] != NULL; i++)
        printf("%s\n", environ[i]);
    
    return 0;
}


事實上,只有execve是真正的系統調用,其它五個函數最終都調用execve,所以execve在man手冊第2節,其它函數在man手冊第3節。這些函數之間的關係如下圖所示。
原文鏈接:https://blog.csdn.net/yychuyu/article/details/80173039

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章