linux exec创建进程,LinuxC创建回收进程fork、exec、wait、waitpid函数的理解

1. fork

int pid = fork();

if (pid == -1 ) {//返回-1,说明fork失败

perror("fork");

exit(1);

} else if (pid > 0) {//返回子进程pid,说明是父进程

} else if (pid == 0) {//返回0,说明是子进程

}

fork出来的子进程和父进程相同的是:全局变量、.data、.text、堆、栈、环境变量、工作目录、宿主目录、信号处理方式等

不同的是:进程id,父进程id,定时器,未决信号集

1-1)父子进程共享文件描述符:

int test_share_fd() {

int fd = open("test_share_fd.txt", O_CREAT | O_RDWR, 0777);//在fork之前open的文件,共享fd以及文件指针offset

if (fd == -1)

{

perror("open file error");

exit(EXIT_FAILURE);

}

int pid = fork();//在fork之后open的文件,共享fd,但不共享文件指针offset

if (pid == -1)

{

close(fd);

perror("fork error");

exit(EXIT_FAILURE);

}

if (pid == 0)

{//child

char* p = "hello,share fd from child";

write(fd, p, strlen(p));

close(fd);

}

else {//parent

wait(NULL);

char buf[512];

lseek(fd, 0, SEEK_SET);

int size = read(fd, buf, sizeof(buf));

buf[size] = 0;

printf("fd from parent:%s\n", buf);

close(fd);

}

return 0;

}

1-2)父子进程共享内存mmap

int test_share_mmap() {

int len = 100;

char* filename = "test_share_mmap_temp";

int fd = open(filename, O_CREAT | O_RDWR, 0777);

if (fd == -1)

{

perror("open file error");

exit(EXIT_SUCCESS);

}

ftruncate(fd, len);

char* p = mmap(NULL, len, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);

if (p == MAP_FAILED)

{

perror("mmap error");

exit(EXIT_FAILURE);

}

int pid = fork();

if (pid == -1)

{

munmap(p, len);

perror("fork error");

exit(EXIT_FAILURE);

}

if (pid == 0)

{

strcpy(p, "hello,share mmap from child");

}

else {

wait(NULL);

printf("mmap from parent:%s\n", p);

}

munmap(p, len);

unlink(filename);

return 0;

}

2. exec

execl,execle用于执行一个可执行文件的,比如test.exe文件

execlp用于执行一个命令行指令的,比如ls -l

int execl(const char *path, const char *arg, ... , NULL);//按路径查找,执行一个文件

//execl("/bin/ls", "ls", "-l", "-a", NULL);

//char* path = "/home/ubuntu/app/test.exe"; execl(path, path, "argv1", "argv2", NULL);

int execle(const char *path, const char *arg, ... , NULL, char * const envp[] */);//同execle

//char* _path = "/home/ubuntu/app/test.exe";

//char* _env[] = { "k1=v1","k2=v2",NULL };使用自定义env,不使用默认的extern char** environ;

//execle(_path, _path, "-l", "-a", NULL, _env);

int execlp(const char *file, const char *arg, ... , NULL);//按PATH查找,执行一个命令

//execlp("ls", "ls", "-l", "-a", NULL);

以下三个exec函数和上面三个类似,上面三个是可变参数传参,以下三个使用数组传参。但是,不管哪种都是需要NULL作为结束符的。

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

int execvp(const char *file, char *const argv[]);

int execvpe(const char *file, char *const argv[],char *const envp[]);

上面六个exec都是linux c函数库的标准函数,在man手册第三章,man 3 execle 即可查看。它们最终都会调用execve这个系统函数,execve函数在man手册第二章。

3. wait,waitpid

孤儿进程:父进程已死,子进程还在,此时子进程就是孤儿进程。子进程的父进程变为init进程,俗称进孤儿院。

僵死进程:父进程还在,子进程已死而且没被回收,这个已死的子进程就是僵死进程。(注意:前提是父进程还在,如果父进程都死了,父子进程都会被回收,就不存在僵死进程了)

僵死进程如何回收?父进程通过wait,waitpid函数回收呀。

pid_t wait(int &status);//阻塞等待子进程退出,回收子进程pcb等残留资源,获取子进程退出状态码

status为传出参数,子进程的退出状态码,该方法堵塞等待。

pid_twaitpid(pid_t pid, int &status, int options);

pid>0:回收指定的子进程,比如pid=1000,表示回收进程id为1000的那个子进程(进程id)

pid=0:回收和调用waitpid同一进程组的任一子进程

pid=-1:回收任一进程组的任一子进程,等价于wait函数

pid

status状态码说明:

if (WIFEXITED(status))//正常退出

{

printf("child exit normal,exit code:%d\n", WEXITSTATUS(status));//正常退出状态码

}

else if (WIFSIGNALED(status))//异常退出

{

printf("child term error,term code:%d\n", WTERMSIG(status));//导致异常退出的状态码

}

else if (WIFCONTINUED(status))//从挂起状态继续运行

{

printf("child continue running");

}

else if (WIFSTOPPED(status))//从运行状态挂起暂停

{

printf("child stop running,stop code:%d\n", WSTOPSIG(status));//导致暂停的状态码

}

options有三个可选值:WCONTINUED,WNOHANG,WUNTRACED

wait是堵塞方法,waitpid可以设置为非堵塞。当options为WNOHANG时为非堵塞,此时waipid返回0表示还没有可回收的子进程,大于0表示回收的子进程。

一次wait或者waitpid调用只能回收一个子进程,所以一般使用do{}while()循环调用。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值