1.exec函数族介绍
函数族:一系列功能相似的函数,类似C++的函数重载,C语言中没有函数重载
exec:并不是生成一个新的进程
一般是先fork一个子进程,然后把子进程的实体替换掉
2.exec的工作原理
只是改变了用户区的数据,内核区的进程id的相关信息并没有改变
3.exec函数族
execl.c
/*
#include <unistd.h>
int execl(const char *path, const char *arg, ...);
- 参数:
- path:需要指定的执行的文件的路径或者名称
a.out /home/nowcoder/a.out 推荐使用绝对路径
./a.out hello world
- arg:是执行可执行文件所需要的参数列表
第一个参数一般没有什么作用,为了方便,一般写的是执行的程序的名称
从第二个参数开始往后,就是程序执行所需要的的参数列表。
参数最后需要以NULL结束(哨兵)
- 返回值:
只有当调用失败,才会有返回值,返回-1,并且设置errno
如果调用成功,没有返回值。
*/
#include <unistd.h>
#include <stdio.h>
int main() {
// 创建一个子进程,在子进程中执行exec函数族中的函数
pid_t pid = fork();
if(pid > 0) {
// 父进程
printf("i am parent process, pid : %d\n",getpid());
// 如果不加sleep,会有孤儿进程的产生导致输出不在一块
sleep(1);
}else if(pid == 0) {
// 子进程
// execl("hello","hello",NULL);
execl("/bin/ps", "ps", "aux", NULL);
perror("execl");
printf("i am child process, pid : %d\n", getpid());
}
for(int i = 0; i < 3; i++) {
printf("i = %d, pid = %d\n", i, getpid());
}
return 0;
}
执行结果:子进程的代码没有执行,因为已经被hello.out替换
hello.c
#include <stdio.h>
int main() {
printf("hello, world\n");
return 0;
}
execlp.c
#include <unistd.h>
int execlp(const char *file, const char *arg, … );
- 会到环境变量中查找指定的可执行文件,如果找到了就执行,找不到就执行不成功。
- 参数:
- file:需要执行的可执行文件的文件名
a.out
ps
- arg:是执行可执行文件所需要的参数列表
第一个参数一般没有什么作用,为了方便,一般写的是执行的程序的名称
从第二个参数开始往后,就是程序执行所需要的的参数列表。
参数最后需要以NULL结束(哨兵)- 返回值: 只有当调用失败,才会有返回值,返回-1,并且设置errno 如果调用成功,没有返回值。 int execv(const char *path, char *const argv[]); argv是需要的参数的一个字符串数组 char * argv[] = {"ps", "aux", NULL}; execv("/bin/ps", argv); int execve(const char *filename, char *const argv[], char *const envp[]); char * envp[] = {"/home/nowcoder", "/home/bbb", "/home/aaa"};
#include <unistd.h>
#include <stdio.h>
int main() {
// 创建一个子进程,在子进程中执行exec函数族中的函数
pid_t pid = fork();
if(pid > 0) {
// 父进程
printf("i am parent process, pid : %d\n",getpid());
sleep(1);
}else if(pid == 0) {
// 子进程
execlp("ps", "ps", "aux", NULL);
printf("i am child process, pid : %d\n", getpid());
}
for(int i = 0; i < 3; i++) {
printf("i = %d, pid = %d\n", i, getpid());
}
return 0;
}