首先先说说进程
1.进程产生
操作运行了一个程序,至少会产生一个进程
问题:进程与程序的区别?
回答:程序就是一个编译好的可执行文件,而进程是操作系统为了执行一个程序而创建的,进程是程序的一次执行过程
二 进程相关的命令
1.ps -ef
查看进程的PID和PPID
2.ps aux
查看进程状态 R:运行态 S:可中断的等待态 D:不能中断的等待态 T:停止态 Z:僵尸态(PID资源未释放)
3.kill
<1>kill -l 查看系统中信号
<2>发送信号到指定的进程
kill -信号 PID
例如:
kill -SIGSTOP 5207
4.killall
给同名的进程发信号
killall -信号 进程名
5.查看系统允许的最大进程个数
cat /proc/sys/kernel/pid_max -> 32768
三 创建子进程
pid_t fork(void);
功能:创建子进程
返回值:
成功给父进程返回子进程的PID,给进程返回0,失败返回-1
思考:如何创建子进程?
回答:复制父进程
问题:为什么创建子进程?
回答:创建子进程的目的是让子进程执行独立任务
问题:子进程和父进程会不会相互影响
回答:不会,父子进程的地址空间是独立
四 父进程回收僵尸态子进程
首先应该知道的概念:
僵尸进程:进程结束,但是父进程没有对它进行收尸处理,即父进程没有对它做资源的回收处理,在内核进程树上还占用一个节点
孤儿进程:父亲进程结束,子进程就会变成孤儿进程,孤儿进程会自动被init进程收养
pid_t wait(int *status);
功能:等待僵尸态子进程
参数:
@status 获取僵尸态子进程状态
返回值:
成功返回僵尸态子进程PID,失败返回-1(没有子进程)
常用用法: wait(NULL);
pid_t waitpid(pid_t pid, int *status, int options);
参数:
@pid -1:等待任意一个子进程退出 子进程的PID:等待指定PID子进程
@status 获取子进程退出的状态
@options 0:阻塞的方式调用 WNOHANG:非阻塞方式调用
返回值:
成功返回僵尸态子进程PID,失败返回-1, 如果没有子进程处于僵尸态,且非阻塞方式调用,此时立即返回,返回值为0
waitpid(-1,NULL,0); => wait(NULL);
waitpid(pid,NULL,0);
waitpid(-1,NULL,WNOHANG);//常用的用法
waitpid(pid,NULL,WNOHANG);
五 结束一个进程
return : 在main函数使用return,用来结束进程
_exit() : 用来结束一个进程
exit() : 用来结束一个进程,刷新缓存
exit(EXIT_SUCCESS) 成功
exit(EXIT_FAILURE) 失败
六:实例如下:
int main(int argc, const char *argv[])
{
int ret;
int status;
nt child_pid1;
int child_pid2;
child_pid1 = fork();
if(child_pid1 <0){
perror("Fail to fork");
return -1;
}
if(child_pid1 == 0){
printf("hello word!\n");
return 0x18;
}
if(child_pid1 > 0){
child_pid2 = fork();
if(child_pid2 <0){
perror("Fail to fork");
return -1;
}
if(child_pid2 == 0){
while(1)
;
}
if(child_pid2 > 0){
while(1)
{
printf("wait ...\n");
ret = wait(&status);
printf("ret : %d\n",ret);
if(ret <0){
printf("Fail to wait : %s\n",strerror(errno));
return -1;
}
if(WIFEXITED(status)){
printf("The child process normally exit , return : %#x\n",WEXITSTATUS(status));
}else if(WIFSIGNALED(status)){
printf("The Child process terminated by signal : %d\n",WTERMSIG(status));
}
}
}
}
return 0;
}
这个程序主要功能为,创建两个子进程,父进程分别等待两个子进程退出,一个为正常退出,一个通过发送信号使其退出
测试结果如下
编译执行:
可以看到子进程1正常退出,用WEXITSTATUS(status)获取了她的返回码
而子进程2一直在轮询,然后,我们查看后台进程
执行 ps aux|grep a.out
可以看到其中一个子进程的确已经退出,然后我们发送信号让另一个子进程退出(你能确定那个是子进程吗?)
kill -SIGKILL 10611 (当然除了SIGKILL ,还可以被其他信号中断(具体执行kill -l),为什么?)
结果如下:
可知进程2已经被信号终止,并且通过WTERMSIG获取了是那个信号
既然wait(&status),可以等待子进程退出,如果换成waitpid呢,请看下面的改造:
测试如下:
可以看出waitpaid(-1,&status,WNOHANGE),会去检测,当前有没有处于僵尸态的子进程,如果没有,立即返回,返回值为0
同理发送信号回终止子进程2
其他用法上面已经说过,就不举例说明!
探究wait与waitpid之间的那些事