进程的正常退出:
1.在main函数中执行 return n;该返回值可以被父进程接收到的,在main函数中与exit几乎等价的
2.进程只有调用了exit函数,就能正常退出,该函数时标准库函数
void exit(int status);
功能:在任何时候调用此函数,都可以结束进程
status:结束状态码
EXIT_SUCCESS
EXIT_FAILURE
效果与main函数中与return的返回值一样
返回值:该函数不会返回
int atexit(void (*function)(void));
功能:注册一个进程结束时要执行的函数
int on_exit(void (*function)(int , void *), void *arg);
功能:注册一个进程结束时要执行的函数
进程退出前要完成:
1.先调用事先注册好的atexit/on_exit函数,注册多个函数时,执行顺序与注册顺序相反
2.冲刷并关闭所有打开状态的IO流
3.该函数的实现调用了_exit/_Exit
3.调用_exit/_Exit
#include <unistd.h>
void _exit(int status);
功能:结束进程,由系统提供
#include <stdlib.h>
void _Exit(int status);
功能:结束进程,由标准库提供
1.它们的参数都会被父进程获取
2.进程结束前会关闭所有打开状态下的文件描述符(系统IO)
3.会向父进程发送SIGCHID信号
4.该函数不会返回
4.进程的最后一个线程执行返回语句时
5.进程的最后一个线程调用了pthread_exit函数
进程的异常终止:
1.调用了abort函数,产生SIGABRT信号
2.进程在执行过程中,接收到某项信号,可以是其他进程发送的,也可能是自己的错误操作导致的
3.进程的最后一个线程收到“取消”操作,并且还做出响应
以上三种结束方式,会让父进程无法获取子进程的结束状态码,故称为异常终止
注意:无论进程是如何结束的,它们最后都会执行同一段代码,关闭所有打开的文件描述符,并释放内存。
子进程回收:
对任何结束方式,都希望能够让父进程知道,通过wait/waitpid函数可以知道子进程是如何结束的,以及获取到结束状态码
#include <sys/wait.h>
pid_t wait(int *status);
功能:等待子进程结束,并获取结束状态码
返回值:结束子进程的ID
1.如果所有的子进程都还在运行,则阻塞
2.如果有一个子进程结束,立即返回该子进程的结束状态码和PID
3.如果没有子进程,则返回-1
WIFEXITED(status) 判断进程是否正常结束,如果是,返回真
WEXITSTATUS(status) 如果进程是正常结束,才可以获取到正确的结束状态码,只获取低八位
WIFSIGNALED(status) 判断进程是否异常结束,如果是,则返回真
WTERMSIG(status) 如何进程是异常结束,可以获取到杀死进程的信号
注意:由于wait函数可能会阻塞,因此不适合在正常的业务逻辑中调用该函数,可以为SIGCHLD信号注册一个信号处理函数,在处理hanshu
中调用wait,因为子进程结束后都会向父进程发送信号
pid_t waitpid(pid_t pid, int *status, int options);
功能:能够指定回收某些或某个进程
pid:
< -1 等待abs(pid)进程组中的进程结束
= -1 等待任意子进程结束,功能与wait等价
= 0 等待同组的任意进程结束
> 0 等待该进程结束
status:结束状态码,与wait等价
options:
WNOHANG 非阻塞模式,如果要等待的进程没有一个结束,立即返回
WUNTRACED 如果有进程处于暂停状态,则返回该进程状态
WCONTINUED 如果有进程从暂停转为继续运行,则返回该进程的状态
WITSTOPPED(status) 判断进程是否处于暂停状态,如果是,则返回真
WSTOPSIG(status) 如果有进程处于暂停状态,则返回导致暂停的信号
WIFCONTINUED(status) 判断该进程是否是由暂停转为继续运行,如果是,返回真
int system(const char *command);
功能:执行一个可执行文件,本质其实是创建了一个子进程去加载该可执行文件
返回值:子进程结束后才返回
该函数的实现,底层调用了fork函数、exec系列函数、waitpid函数,其实是让进程创建了一个子进程,然后让子进程去加载了command可执行文件