【Linux】进程控制-----进程创建与进程终止
目录
前言:
在将进程创建之后还要对其进行合理的管理,使用进程控制相关的知识对进程进行管理就能够解决很多其他问题
一、进程创建:
1、fork函数
fork函数就是在当前进程中创造一个新进程,这个新进程就是当前进程的子进程,当前进程就是新进程的父进程,子进程的代码和数据就是剩余父进程的代码和数据
返回值:
这个返回值的类型是pid_t这个是typedef int,使用与进程的,fork函数会有两个返回值:
创建成功:子进程返回0,父进程返回子进程的pid
创建失败:返回-1
为什么要给子进程返回0,父进程返回子进程的pid
这样有两个返回值是为了区分子进程和父进程,我们不妨想想,为什么要有子进程和父进程呢?这样的原因主要是为了让子进程执行和父进程不一样的代码块,所以就需要不同的返回值来区分子进程和父进程。
这样就可以用if 、else if、 else进行分流操作,来完成子进程完成和父进程不一样的操作
并且一个子进程是只有一个父进程的,一个父进程可能有许多子进程,所以就需要给子进程一个返回0来标记,给父进程返回子进程的PID来管理子进程。
还有其他几个问题在下面文章中所写了,如
1、fork函数究竟干了什么?
2、一个函数是如何做到返回两次的,这怎么理解?
3、同一个变量怎么会有不同的内容,如何理解?
2、创建多个进程:
如上代码就是使用for循环创建多个进程,最后的sleep(1000)是不让父进程结束,如果父进程结束了就会使子进程变成孤儿进程,然后在调用函数child里面打印子进程的pid和ppid进行观察
如上,左边就是每次子进程的调用,可以看到,子进程的调用是毫无规律的,也就是说在多进程的调用中,哪一个进程被调用是不知道的,这是由调度器决定的,看调度器把哪一个进程先放入调度队列中谁就先进行调度
3、写时拷贝:
写时拷贝实际上是通过页表的方式,对不同的进程进行空间寻址,达到出现改写行为时,父子进程指向的数据就会不同的机制
当进行写时拷贝前:
当进行写时拷贝后:
如上,无论是父进程还是子进程进行数据的修改都会进行写时拷贝,就会在物理空间中开辟一块新空间,然后就会重新映射物理地址到新开辟的空间,这就是写时拷贝原理
二、进程终止:
如果进程终止的话,那么就必然是下述的三种情况
代码运行完毕,结果正确
代码运行完毕,结果不正确
代码异常终止(进程崩溃)
进程退出码:
在命令行解释器中可以用
echo $?来进行查看上一个进程的退出码
当我们的代码运行起来就变成了进程,当进程结束后main函数的返回值实际上就是该进程的进程退出码,这个退出码对应着相应的退出信息,所谓的退出信息就是以退出码的形式作为main函数的返回值返回,这样就能够知道是因为 什么而导致进程崩溃,一般main函数的退出码是0的话就是正常运行结束,如果不是0的话就是异常退出,
这样echo看到的就是正常结束
strerror函数可以通过打印出退出码对应的信息:(这个函数在string文件中)
在Linux中有133个退出码(不同环境下退出码的个数和对应的含义也可能会不同)
如下:当进程中被热键ctrl + c终止后对应的退出码就是130在上面看到就是 自我退出 的
如上,如果在进行echo $?的话就是看的echo这个进程的退出码了,所以看一个指令的退出码,一般都是0,
我们也可以模拟实现一个错误码
这样的话也可以实现错误码(数组下标)对应的错误信息了
退出方式:
1、return退出
一般return退出就是退出当前函数到上一级
2、热键退出(ctrl+c)
如果是热键退出的话就属于外部退出,这样退出的话在Linux中就属于130号退出码,
返回码137通常表示命令或进程被强制终止,具体原因包括以下几种,命令被另一个程序或脚本通过发送SIGKILL信号终止:在Linux系统中,SIGKILL信号(值为9)是一个强制终止进程的信号
3、exit与_exit
这两个退出函数,本质上来说,没有区别,都是退出进程,但是exit()是对_exit()做的封装实现,推荐一般使用exit毕竟这个函数还会多做一点事比如冲刷缓冲区等等
看看如下代码:
运行结果是如下的:
原因:exit是终止进程的,当执行完exit之后就会结束这个进程就不会继续往下进行了
_exit与exit区别:
当执行如下的代码的时候,就会刷新缓冲区,这样的话就可以看见所打印的内容了
同样的代码当是 _exit 的时候就看不到所打印的内容
原因:
exit()是对_exit()做的封装实现
_exit()就只是单纯的退出程序
而exit()在退出之前还会做一些事,比如冲刷缓冲区,再调用 _exit()
进程异常退出:
当进程异常退出的时候,再看退出码就没有意义了
在Linux中,进程异常退出可能有以下几种情况:
接收到信号导致的异常退出:常见的信号如kill -9 ,热键ctrl + c
资源耗尽:例如,CPU时间、内存等资源用尽
程序代码错误:例如,数组越界、无效指针引用等
系统调用错误:进程执行了非法的系统调用
缓冲区:
由于_exit函数是一个系统调用,直接由操作系统内核提供,所以如果缓冲区也在内核中的话,那么在_exit结束进程的时候就可以刷新缓冲区了,但是_exit却不能够刷新缓冲区,所以缓冲区不在内核中
原文地址:https://blog.csdn.net/2303_80828380/article/details/144026260
免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!