进程控制
一、进程控制
1.1、fork函数
在linux中fork函数就是在已存在的进程中创建一个新进程。新进程为子进程,原进程为父进程。
重点:fork函数的返回值为0代表的是子进程,大于0代表的是父进程,小于0代表的是创建子进程失败。
在进程调用fork后,控制转移到内核中的fork代码后,内核接下来会做什么?
- 分配新的内存块和内核数据结构给子进程。
- 将父进程部分数据结构内容拷贝至子进程。
- 添加子进程到系统进程列表当中。
- fork返回,开始调度器调度。
注意:在fork之前父进程是独立执行的,在fork之后父子进程的执行流分别进行执行。
1.2 写时拷贝
在创建子进程之后,父子代码共享,如果父子进程都不在写入时,数据也是共享的,但是当任意一方进行写入时,那么以写时拷贝的方式各自拷贝一份副本。
二、进程终止
2.1进程退出场景
- 代码运行完毕,结果正确;
- 代码运行完毕,结果不正确;
- 代码异常终止;
2.2进程退出方法
正常终止:
- 从main函数返回;
- 调用exit函数
- _exit
_exit和exit之间的却别:
#include <unistd.h>
void exit(int status);
#include <unistd.h>
void _exit(int status);
参数:status 定义了进程的终止状态,父进程通过wait来获取该值
异常退出:
ctrl+c,进行信号终止。
return 退出
return是一种更为常见的退出进程的方法,执行return n等同于执行exit(n),因为调用main运行时函数会将main的返回值当作exit的参数。
三、进程等待
3.1 进程等待的必要性
子进程退出,如果父进程不管不顾,那么就会造成“僵尸进程”,进而造成内存泄漏,进程如果是“僵尸进程”那么kill-9进程也无法进行处理,因为没有办法杀掉一个已经死去的进程,而且在正常进行中,父进程需要知道子进程任务完成的情况,父进程需要通过进程等待的方式回收子进程资源,并且获取子进程退出信息。
3.2 进程等待的方法
wait方法
#include<sys/types.h>
#include<sys/wait.h>
pid_t wait(int*status);
返回值:
成功返回被等待进程pid,失败返回-1。
参数:
输出型参数,获取子进程退出状态,不关心则可以设置成为NULL
waitpid方法
pid_ t waitpid(pid_t pid, int *status, int options);
返回值:
当正常返回的时候waitpid返回收集到的子进程的进程ID;
如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;
参数:
pid:
Pid=-1,等待任一个子进程。与wait等效。
Pid>0.等待其进程ID与pid相等的子进程。
status:
WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)
options:
WNOHANG: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进
程的ID。
调用wait/waitpid后三种情况
- 如果子进程已经退出,调用wait/waitpid时,wait/waitpid会立即返回,并且释放资源,获得子进程退出信息。
- 如果在任意时刻调用wait/waitpid,子进程存在且正常运行时,则进程可能阻塞。
- 如果不存在子进程,则会立即出错返回。