目录
进程终止
进程退出场景
1.正常退出:从main函数返回、调用exit()、调用_exit()。
2.异常退出:ctrl+c、信号终止、空指针访问等。
exit与_exit
1.exit()函数是用来终止一个进程的,谁调用终止谁:
例如以下代码,使用exit()函数来终止一个进程:
可以看到,代码执行后只打印出了x的值,exit()函数调用后进程就正常退出了。
2._exit()函数也是用来终止一个进程的,谁调用终止谁:
如下代码,使用_exit()函数来终止一个进程:
从以上代码可以看出,调用完_exit()函数后进程就正常退出了。
exit与_exit的区别与联系
1.exit是C标准库的函数,而_exit是系统调用函数。
2.exit最后也会调用_exit,但是在这之前会做两件事:如下图
执行用户自定义的清理函数
自定义一个清理函数,然后通过atexit()函数将清理函数注册为回调函数, 这样在进程退出时就会自动调用了,如下:
如果将exit()换成_exit(),那么以上代码中自定义的函数将不会被调用。
冲刷缓冲区,关闭流
缓冲区时C标准库定义的,并不属于操作系统内核。建立缓冲区是为了减少IO次数,IO操作比较耗费时间,当触发冲刷缓冲区的条件后,缓冲区的内容才会进行IO操作(打印到屏幕、写入文件等)。
所以说调用_exit函数并不会冲刷缓冲区,看如下代码:
那么如何能冲刷缓冲区呢?
1.调用exit函数,将以上代码中的_exit换成exit就可以触发冲刷缓冲区的条件了。
2."\n"也会冲刷缓冲区:
3. 调用fflush()函数手动刷新:
4.main函数的return也会刷新缓冲区:
进程等待
什么是进程等待
wait方法
wait()是进程等待时调用的一个函数,下面来介绍这个函数:

那么使用wait后父进程是否会在子进程退出前退出呢?
可以使用命令:pstack [PID] 来查看父进程在做什么,输入命令后:
通过以上代码说明了wait()函数的阻塞特性。
总结一下阻塞特性含义:如果发起阻塞调用的时候资源在,无需等待,执行完函数功能后返回。如果发起阻塞调用的时候资源不在,等待资源到来后,执行完函数功能后返回。
waitpid方法
waitpid函数功能和wait函数功能类似,但是可以输入要等待的子进程PID,也可以设置为非阻塞。 下面一段代码主要验证waitpid函数被设置为非阻塞时的特点:
设置为非阻塞后,子进程并没有被等待到,如何解决以上问题?
对于以上代码,waitpid()函数的调用可以搭配循环,这样就可以等待到子进程了:
总结一下非阻塞特性的含义: 如果发起阻塞调用的时候资源在,执行完函数功能后返回。如果发起阻塞调用的时候资源不在,直接返回0。
所以说非阻塞调用需要搭配循环来使用。
参数status的含义
status是int* 类型的,可以通过外部传递整型变量的地址,拿到子进程的退出信息,分为两种情况:
1.子进程正常退出:会获取到子进程的退出码。
2.子进程异常退出:会获取到coredump标志位+退出信号
但是获取到的退出信息的整型变量只用到了四字节中的前两个地址较低的字节,如下图:
子进程正常退出时,退出码被设置,coredump标志位为0,没有退出信号。如下代码:
子进程异常退出时,coredump标志位为1,退出信号被设置。
首先使用命令设置核心转储文件的大小为:unlimited,也就是不限制。
再看如下代码:
子进程异常退出,coredump标志位为1,退出信号被设置,没有退出码。