目录
一、进程创建
1.fork函数
2.写实拷贝
二、进程终止
1.进程退出场景
2.进程常见退出方法
(1)从main函数返回
(2)_exit
(3)调用exit
三、进程等待
1.进程等待的方法
(1)wait方法
(2)waitpid方法
2.获取子进程的status
一、进程创建
1.fork函数
#include <unistd.h>
pid_t fork(void);
//返回值:子进程中返回0,父进程返回子进程的id,出错返回-1
- 代码实例:
#include <stdio.h>
#include <unistd.h>
int main()
{pid_t pid;printf("Before: pid is %d\n", getpid());if((pid = fork()) == -1 ) perror("fork()"),exit(1);printf("After:pid is %d, fork return %d\n", getpid(), pid);sleep(1);return 0;
}
- 运行结果:
- 此时内部指向如下:
fork 之前父进程独立执行, fork 之后,父子两个执行流分别执行
2.写实拷贝
通常,父子代码共享,父子再不写入时,数据也是共享的,当任意一方试图写入,便以写时拷贝的方式各自一份副本。具体见下图:
二、进程终止
1.进程退出场景
- 代码运行完毕,结果正确
- 代码运行完毕,结果不正确
- 代码异常终止
2.进程常见退出方法
(1)从main函数返回
- 查看进程退出码的方式: $?
- 当退出码为0时表示成功退出;退出码不为0时表示异常退出,不同数字表示不同错误原因
- 1:表示一般性的错误或命令未能找到指定的文件或目录
- 2:表示用法错误或命令的某些参数不正确
- 126:表示命令不可执行
- 127:表示命令未找到
- 128及以上:通常表示由信号引起的退出
(2)_exit
#include <unistd.h>
void _exit(int status);
//status定义了进程终止状态,表示退出码
(3)调用exit
#include <unistd.h>
void _exit(int status);
//status定义了进程终止状态,表示退出码
exit 最后也会调用 exit, 但在调用 exit 之前,还做了其他工作:
- 执行用户通过 atexit或on_exit定义的清理函数。
- 关闭所有打开的流,所有的缓存数据均被写入
- 调用_exit
三、进程等待
1.进程等待的方法
(1)wait方法
#include <sys/types.h>
#include <sys/wait.h>pid_t wait(int *status);//返回值:成功则返回被等待进程得pid;失败则返回-1
//参数:输出型参数,获取子进程退出状态,不关心则可设置为NULL
(2)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。
2.获取子进程的status
- wait和waitpid,都有一个status参数,该参数是一个输出型参数,由操作系统填充,传递NULL,表示不关心子进程的退出状态信息,操作系统会根据该参数,将子进程的退出信息反馈给父进程
- status不能简单的当作整形来看待,可以当作位图来看待