小白杨的博客

这是一个记录学习的博客

linux操作系统之进程exec函数族

fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后该进程的id并未改变。
将当前进程的.text、.data替换为所要加载的程序的.text、.data,然后让进程从新的.text第一条指令开始执行,但进程ID不变,换核不换壳。
其实有六种以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判断。
l (list) 命令行参数列表
p (path) 搜素file时使用path变量
v (vector) 使用命令行参数数组
e (environment) 使用环境变量数组,不使用进程原有的环境变量,设置新加载程序运行的环境变量

execlp函数
加载一个进程,借助PATH环境变量
int execlp(const char *file, const char *arg, …); 成功:无返回;失败:-1
参数1**:要加载的程序的名字。**该函数需要配合PATH环境变量来使用,当PATH中所有目录搜索后没有参数1则出错返回。
该函数通常用来调用系统程序。如:ls、date、cp、cat等命令。

#include<unistd.h>
#include<stdlib.h>
#include<stdio.h>
int main()
{
        pid_t pid;
        pid = fork();
        if(pid==-1)
        {
                perror("fork.");
                exit(1);
        }
        else if(pid>0)
        {
                printf("i am parent.\n");
        }
        else
        {
                execlp("ls","ls","-l","-a",NULL);
        }
        return 0;
}

execl函数
加载一个进程, 通过 路径+程序名 来加载。
int execl(const char *path, const char *arg, …); 成功:无返回;失败:-1
对比execlp,如加载”ls”命令带有-l,-F参数
**execlp(“ls”, “ls”, “-l”, “-F”, NULL); 使用程序名在PATH中搜索。
execl(“/bin/ls”, “ls”, “-l”, “-F”, NULL); 使用参数1给出的绝对路径搜索。**

#include<unistd.h>
#include<stdlib.h>
#include<stdio.h>
int main()
{
        pid_t pid;
        pid = fork();
        if(pid==-1)
        {
                perror("fork.");
                exit(1);
        }
        else if(pid>0)
        {
                printf("i am parent.\n");
        }
        else
        {
                execl("/bin/ls","ls","-l","-a",NULL);
                execl("./myapp","./myapp",NULL);//调用自定义程序
        }
        return 0;
}

execvp函数
加载一个进程,使用自定义环境变量env
int execvp(const char *file, const char *argv[]);
变参形式: ①… ② argv[] (main函数也是变参函数,形式上等同于 int main(int argc, char *argv0, …))
变参终止条件:① NULL结尾 ② 固参指定
execvp与execlp参数形式不同,原理一致。
练习:将当前系统中的进程信息,打印到文件中。

#include<unistd.h>
#include<stdlib.h>
#include<stdio.h>
#include<sys/types.h>
#include<fcntl.h>

int main()
{
        int fd;
        fd = open("mytest.out",O_WRONLY|O_CREAT|O_TRUNC,0777);
        if(fd<0)
        {
                perror("fopen:");
                exit(1);
        }
        dup2(fd,STDOUT_FILENO);//dup2(3,1);

        execlp("ps","ps","ax",NULL);
        close(fd);
        return 0;
}
阅读更多
文章标签: 进程 exec
上一篇gdb对进程的调试
下一篇linux操作系统之进程 僵尸进程和孤儿进程
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

关闭
关闭