一、exec函数族原理
fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种exec函数以执行另一个程序。
当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新的程序替换,从新程序的启动例程开始执行。
调用exec并不创建新进程,所以调用exec前后该进程的id并未改变。
将当前进程的.text、.data替换为索要加载的程序的.text、.data,然后让进程从新的.text第一条指令开始执行,但进程ID不变,换核不换壳
共有6种以exec开头的函数,统称为exec函数:
其中,

重点掌握前两个execlp和execl函数
二、execlp和execl函数
1,execlp函数:加载一个进程,借助PATH变量
execlp()会从PATH 环境变量所指的目录中查找符合参数file 的文件名, 找到后便执行该文件, 然后将第二个以后的参数当做该文件的argv[0], argv[1], …, 最后一个参数必须用空指针(NULL)作结束.

参数1:要加载的程序的名字。该函数需要配合PATH环境变量来使用,当PATH中所有目录搜索后没有参数1则出错返回
该函数通过用来调用系统程序,如ls、date、cp、cat等命令
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 | //ls -l-h #include<stdio.h> #include<stdlib.h> #include<string.h> #include<unistd.h> #include<pthread.h> int main(int argc, char *argv[]) { pid_t pid = fork();//创建子进程 if(pid == -1) { perror("fork error") exit(1); }else if(pid == 0)//子进程 { //错误写法execlp("ls","-l","-h",NULL);//NULL是设立的哨兵必须手动加 execlp("ls","ls","-l","-d","-h",NULL); //只有执行错误才会有返回,一旦执行成功就会去执行ls这个文件,就不回来了 perror("exec error");//这两行代码之后上行代码出错的时候才有机会运行 exit(1); }else if(pid>0)//父进程 { sleep(1);//为了让子进程先执行父进程后执行 printf("i am parent : %d\n",getpid()); } return 0; } |

2,execl函数
改函数名与参数即可

或者

第一个参数是文件路径,第二个参数是argv[0],之后是传给argv[0]的参数,没有了就哨兵NULL
3,execlp执行的是系统的文件,execl执行的是自己的文件,当然也可以用execl执行系统的文件,写成绝对路径就行
三、execvp函数,v是argv参数的v

将参数进行了一个封装
四、exec函数族一般规律
exec函数一旦调用成功即执行新的程序,不返回。只有失败才返回-1,所以通过可以直接在exec函数后直接调用perror()和exit(),不需要if判断