Linux进程控制(上)

目录

进程终止

进程退出场景

exit与_exit

exit与_exit的区别与联系

  执行用户自定义的清理函数

冲刷缓冲区,关闭流

 进程等待

什么是进程等待

wait方法

waitpid方法

参数status的含义


进程终止

进程退出场景

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()函数的功能,如下代码中子进程不会出现僵尸状态:

 那么使用wait后父进程是否会在子进程退出前退出呢?

可以使用命令:pstack [PID] 来查看父进程在做什么,输入命令后:

 通过以上代码说明了wait()函数的阻塞特性。

总结一下阻塞特性含义:如果发起阻塞调用的时候资源在,无需等待,执行完函数功能后返回。如果发起阻塞调用的时候资源不在,等待资源到来后,执行完函数功能后返回。

waitpid方法

waitpid函数功能和wait函数功能类似,但是可以输入要等待的子进程PID,也可以设置为非阻塞。 下面一段代码主要验证waitpid函数被设置为非阻塞时的特点:

设置为非阻塞后,子进程并没有被等待到,如何解决以上问题?

对于以上代码,waitpid()函数的调用可以搭配循环,这样就可以等待到子进程了:

 总结一下非阻塞特性的含义: 如果发起阻塞调用的时候资源在,执行完函数功能后返回。如果发起阻塞调用的时候资源不在,直接返回0。

所以说非阻塞调用需要搭配循环来使用。

参数status的含义

status是int* 类型的,可以通过外部传递整型变量的地址,拿到子进程的退出信息,分为两种情况:

1.子进程正常退出:会获取到子进程的退出码。

2.子进程异常退出:会获取到coredump标志位+退出信号

但是获取到的退出信息的整型变量只用到了四字节中的前两个地址较低的字节,如下图:  

子进程正常退出时,退出码被设置,coredump标志位为0,没有退出信号。如下代码:

 子进程异常退出时,coredump标志位为1,退出信号被设置。

首先使用命令设置核心转储文件的大小为:unlimited,也就是不限制。

 再看如下代码:

子进程异常退出,coredump标志位为1,退出信号被设置,没有退出码。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值