程序员的成长之路

互联网/程序员/技术/资料共享

关注

阅读本文大概需要 5.8 分钟。

来自:https://juejin.im/post/5f20fbeae51d45348675fa78

那时刚写公众号,当时记录的学习笔记,现在看来,之前记录的有一个错误的地方,当时也没察觉到。

写错了就要改嘛,程序员也不能怕错~

不知道大家看自己几年前的做事情,有的时候有没有一种感觉,这是我做的吗???

好吧,有点嫌弃当时的自己~

直接进入正题吧,父子进程之间到底有啥关系?

进程

先来说下什么是进程:

来看下百度是怎么说的:

光看说的不够形象,在windows系统中,它长这样:

在Mac系统中,它长这样:

Linux中是这样的:(有点长截图一部分好了)

[root@iz2ze76ybn73dvwmdij06zz ~]# ps -ef
UID        PID  PPID  C STIME TTY          TIME CMD
root         1     0  0 5月20 ?       00:00:33 /usr/lib/systemd/systemd --system --deserialize 21
root         2     0  0 5月20 ?       00:00:00 [kthreadd]
root         3     2  0 5月20 ?       00:00:06 [ksoftirqd/0]
root         5     2  0 5月20 ?       00:00:00 [kworker/0:0H]
root         7     2  0 5月20 ?       00:00:02 [migration/0]
root         8     2  0 5月20 ?       00:00:00 [rcu_bh]
root         9     2  0 5月20 ?       00:30:40 [rcu_sched]
root        10     2  0 5月20 ?       00:00:17 [watchdog/0]
root        11     2  0 5月20 ?       00:00:16 [watchdog/1]
root        12     2  0 5月20 ?       00:00:02 [migration/1]
root        13     2  0 5月20 ?       00:00:03 [ksoftirqd/1]
root        15     2  0 5月20 ?       00:00:00 [kworker/1:0H]
root        17     2  0 5月20 ?       00:00:00 [kdevtmpfs]
root        18     2  0 5月20 ?       00:00:00 [netns]
root        19     2  0 5月20 ?       00:00:01 [khungtaskd]
root        20     2  0 5月20 ?       00:00:00 [writeback]
root        21     2  0 5月20 ?       00:00:00 [kintegrityd]
root        22     2  0 5月20 ?       00:00:00 [bioset]
root        23     2  0 5月20 ?       00:00:00 [kblockd]

OK,以上每一行都是对一个进程的描述,来具体看一下每个参数的含义:

标示

描述

UID

用户ID

PID

进程ID

PPID

父进程ID

C

进程占cpu百分比

STIME

进程启动的时间

TTY

终端机位置

TIME

实际使用cpu的时间

CMD

命令以及参数

我们现在知道了每个参数的含义,既然讲到进程嘛,首先,进程ID是唯一的并且是非负数的,但是进程ID是可以复用的,毕竟进程也会终止。

可以看到没有进程PID是0的,这是为什么呢?黑人问号脸?

0一般来说是系统进程,属于内核的一部分,不执行任何磁盘上的程序。

fork

一个进程可以通过调用fork函数创建新的进程,被创建出来的这个进程就叫子进程。

这里需要注意一下,fork函数的返回值父子进程区别。

  • 子进程 :返回值是0,返回0的理由是子进程的父进程是可以唯一确定的,通过getppid方法可以获取到父进程id。

  • 父进程 : 返回的是新创建的子进程的id,因为父进程可以有多个子进程,也没有这样的函数可以获取该线程的子线程的所有id。

下边的话我们来验证一下上说的这一段话。准备好脚本。

#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>int main(int argc, char const *argv[])
{pid_t p1 = fork();printf("%d\n",p1);if(p1 > 0){printf("父进程 pid = %d, p1 = %d\n", getpid(), p1);}else{printf("子进程 pid = %d , ppid = %d, p1 = %d\n", getpid(), getppid(), p1);}return 0;
}

运行看结果:

[root@iz2ze76ybn73dvwmdij06zz ~]# ./fork2
10213
父进程 pid = 10212, p1 = 10213
0
子进程 pid = 10213 , ppid = 10212, p1 = 0

通过上面的小例子我们可以看到父进程的返回值是子进程的ID,子进程的返回是0。

孤儿进程

孤儿我们都懂就是。。。

是的,没错孤儿进程也是这样的,就是没有父进程的进程。当然创建的时候肯定是要先创建父进程了,当父进程退出时,它的子进程们(一个或者多个)就成了孤儿进程了。

接下来在继续做一个小测试,让父进程现退出。准备好脚本。

[root@iz2ze76ybn73dvwmdij06zz ~]# cat guer.c
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>int main()
{pid_t pid = fork();if (pid < 0) {perror("fork error;");exit(1);} else if (pid == 0) {sleep(5);printf ("子进程 : [ pid] = %d , 父进程 [ppid] = %d\n",getpid(),getppid());exit(0);} else if (pid > 0) {printf("我是父线程,我先退出一步~\n");exit(0);}return 0;
}

执行并看结果:

到这里估计很多童鞋估计已经看懂了,父进程退出后,子进程被一个进程ID为1的进程领养的。还挺好这个结果,至少还是有人管的,被暖到了~ 进程id为1的进程是init进程,每当有孤儿进程出现时,init进程就会收养它并成为它的父进程 ,来照顾它以孤儿进程以后的生活。

危害

因为孤儿进程会被init进程接管,所以孤儿进程是没有危害的。

僵尸进程

和孤儿进程相反的是,这次是子进程先退出,而父进程又没有去处理回收释放子进程的资源,这个时候子进程就成了僵尸进程。

先准备好代码:

[root@iz2ze76ybn73dvwmdij06zz ~]# cat zombie.c#include <stdio.h>#include <unistd.h>#include <errno.h>#include <stdlib.h>int main(){pid_t pid;pid = fork();if (pid < 0){perror("fork error:");exit(1);}else if (pid == 0){printf("我是子进程,我要先退出一步了.\n");printf("子进程 id : %d\n" ,getpid());exit(0);} else {printf("我是父进程,我先睡2秒\n");printf("父进程 id : %d\n" ,getpid());sleep(2);while(2); //来个死循环,不退出的那种}return 0;}

运行看下结果:

再来开一个终端看下进程结果:

大家可以看到,子进程并没有完全退出,释放资源,而是变成了僵尸进程。

危害

资源上是占用不了什么资源。但是通常系统的进程数量都是有限制的,如果有大量的僵尸进程占用进程号,导致新的进程无法创建,这个危害类似于占个坑,不办事。

处理

1.干掉父进程

干掉父进程后,让剩下的子进程成为孤儿进程,成为孤儿进程后就和我们上面说的一样了,由init进程来领养这些进程,并且来处理这些进程的资源释放等工作。

2.父进程调用wait或waitpid

等函数等待子进程结束,这会导致父进程挂起。执行wait()或 waitpid()系统调用,则子进程在终止后会立即把它在进程表中的数据返回给父进程,此时系统会立即删除该进入点。在这种情形下就不会产生defunct进程。

3.fork两次

第一次 fork : 父进程fork一个子进程

第二次 fork : 子进程fork一个孙进程后退出

那么孙进程被init接管,当孙进程结束后,init会回收。

但子进程的回收还要自己做。

4.signal函数

父进程来处理:用signal函数为SIGCHLD安装handler,在子进程结束后,父进程会收到该信号,可以在handler中调用wait回收。

内核来处理: 如果父进程不关心子进程什么时候结束,可以通过以下两个函数通知内核自己不感兴趣子进程的结束,此时,子进程结束后,内核会回收并不再给你父进程发信号。

  • signal(SIGCLD, SIG_IGN)

  • signal(SIGCHLD, SIG_IGN)

总结

本来以为简单的一个问题,没想到这个篇幅其实也不算短,所以感觉程序员真的不能说一个什么知识点就简单,很容易理解啊,一旦你想要深入也是需要一定的时间花费和精力的~

<END>

推荐阅读:

互联网人办公地鄙视链

从零开始搭建创业公司后台技术栈

5T技术资源大放送!包括但不限于:C/C++,Linux,Python,Java,PHP,人工智能,单片机,树莓派,等等。在公众号内回复「2048」,即可免费获取!!

微信扫描二维码,关注我的公众号

写留言

朕已阅 

有两个这样的进程:僵尸进程孤儿进程,蓝瘦香菇相关推荐

  1. 启动进程 问号_有两个这样的进程:僵尸进程amp;孤儿进程,蓝瘦香菇

    进程 先来说下什么是进程: 来看下百度是怎么说的: 光看说的不够形象,在windows系统中,它长这样: 在Mac系统中,它长这样: Linux中是这样的:(有点长截图一部分好了) [root@iz2 ...

  2. delphi pid判断进程结束_有两个这样的进程:僵尸进程amp;孤儿进程,蓝瘦香菇

    进程 先来说下什么是进程: 来看下百度是怎么说的: 光看说的不够形象,在windows系统中,它长这样: 在Mac系统中,它长这样: Linux中是这样的:(有点长截图一部分好了) [root@iz2 ...

  3. day34 并行并发、进程开启、僵尸及孤儿进程

    day34 并行并发.进程开启.僵尸及孤儿进程 1.并行与并发 什么是并行? 并行指的是多个进程同时被执行,是真正意义上的同时 什么是并发? 并发指的是多个程序看上去被同时执行,这是因为cpu在多个程 ...

  4. 僵尸进程zombie与孤儿进程orphan

    代码已上传至https://github.com/gatieme/AderXCoding/tree/master/system/unix/zombie 问题提出 以前在学习<unix环境高级编程 ...

  5. 【进程控制(进程退出、孤儿进程、僵尸进程)_Linux】

    01 / 进程退出 02 / 孤儿进程 03 / 僵尸进程 04 / 进程回收 wait()函数 退出信息相关宏函数 waitpid()函数 01 / 进程退出 //标准C库函数 #include & ...

  6. Linux进程学习(孤儿进程和守护进程)

    孤儿进程和守护进程 通过前面的学习我们了解了如何通过fork()函数和vfork()函数来创建一个进程.现在 我们继续深入来学习两个特殊的进程:孤儿进程和守护进程 一.孤儿进程 1.什么是 孤儿进程 ...

  7. Linux中的进程控制:进程退出、孤儿进程、僵尸进程 概念及代码示例 [Linux高并发服务器开发]

    目录 一.进程退出 二.孤儿进程 三.僵尸进程 一.进程退出 #include <stdlib.h> void  exit ( int status ); #include <uni ...

  8. 僵尸和孤儿进程及虚拟内存

    调研进程的调度算法. 根据系统的资源分配策略所规定的资源分配算法.对于不同的的系统和系统目标,通常采用不同的调度算法,例如,在批处理系统中,为了照顾为数众多的段作业,应采用短作业优先的调度算法:又如在 ...

  9. 进程退出、孤儿进程、僵尸进程

    exit() /* //Linux 系统库 #include <unistd.h> void _exit(int status);-参数:status:进程退出时的状态信息,父进程挥手子进 ...

最新文章

  1. 70.nodejs操作mongodb
  2. 看完这篇!Linux网络基础知识通关!
  3. ASP.NET Core Web Razor Pages系列教程一:使用ASP.NET Core 创建一个Razor Pages网络应用程序
  4. 机器学习中对抗性攻击的介绍和示例
  5. JCo3 建立连接到SAP(2)- 连接池
  6. 成为指标的“绝地武士”:tableau创建指标的 10 个技巧和窍门
  7. Linux 0.11 实验环境搭建
  8. 企业的核心竞争力是什么
  9. java for 循环执行顺序
  10. 学java要算法吗_学习java不可不知的几种算法
  11. Siri为什么越来越蠢?
  12. 物理实验模拟软件_网络系统实验平台:发展现状及展望
  13. 剑指offer——面试题24:二叉搜索树的后序遍历序列
  14. 何万青 | 从天河2号到阿里云超算,P9技术大牛的职业发展智慧
  15. 双击IE出现打开方式解决办法
  16. 钽电容技术参数及封装
  17. MATLAB做三维图时值为0的点不画出来
  18. linux定时清理临时目录,tmp临时文件目录自动清理
  19. AIBU-在建工程转固定资产(预转固)报错:消息号AW002 资产无单项需结算
  20. SpringBoot日志框架篇

热门文章

  1. 喜剧研究一[武林外传]
  2. 第三周OJ刷题渊子赛马
  3. Android 打开手机QQ,实现类似于客服功能
  4. springboot整合apidoc
  5. container[prop]
  6. 故事的魅力—读王小波的《绿毛水怪》
  7. 苹果自研M1笔记本芯片投入65亿开发自研基带。
  8. python中凯撒密码加密_凯撒密码加密Python
  9. Go 1.18 发布了,不用翻墙即可下载!!!
  10. matlab GUI 打包程序(Application Complier生成exe文件和App打包)