当前位置:   article > 正文

【Linux操作系统】:Linux进程概念(2)_僵尸进程 linux

僵尸进程 linux

目录

一、Z(zombie)-僵尸进程

1.僵尸进程概念

 2.僵死状态(进程处理)

3.僵尸进程的危害

4.僵尸进程的模拟实现

二、孤儿进程 

1.孤儿进程的概念

2.模拟实现孤儿进程

三、进程优先级

1.进程优先级基本概念

2.查看系统进程

 3.PRI and NI

4.查看进程优先级的命令(top)

5.为啥要有优先级 

四、环境变量

1.环境变量基本概念

​编辑2.为什么需要环境变量

3.常见的环境变量 

4.查看环境变量的方法 

​编辑

5.测试PATH

6.环境变量的组织方式

7.和环境变量相关的命令

8.通过代码如何获取环境变量

1.命令行三个参数

2.通过第三方变量environ获取

3.系统调用getenv和putenv函数

9.本地变量和环境变量以及内键命令


一、Z(zombie)-僵尸进程

1.僵尸进程概念

故事

   张三每天都有跑步的习惯,这一天他和往常一样跑步,跑了两三圈,突然跑在它前面的一个人倒在地上不动了,作为热心市民张三赶紧报警并且拨打120。很快120就来了,但是没过几分钟就走了,这个人已经噶了(去世了),并交给了警察。警察这个时候会怎样做呢?警察立刻封锁了现场,并且叫来法医,法医确认了这个人的死亡原因,并且收集了所有信息。然后告诉家属信息。这个时候再把这个人抬走。

 请问从这个人倒下到他被抬走,他是死亡状态吗?

不是,他是死亡了,但是他不是死亡状态,这个时候他是僵死状态,直到他被抬走他才是死亡状态。

  • 僵死状态(Zombies)是一个比较特殊的状态。当进程退出并且父进程(使用wait()系统调用) 没有读取到子进程退出的返回代码时就会产生僵死(尸)进程。
  • 僵死进程会以终止状态保持在进程表中,并且会一直在等待父进程读取退出状态代码。
  • 所以,只要子进程退出,父进程还在运行,但父进程没有读取子进程状态,子进程进入Z状态

 


 2.僵死状态(进程处理)

解决方法

  1. 重启操作系统(建议不推荐)。
  2. 利用信号SIGCHLD,子进程状态改变后会向其父进程发送SIGCHLD信号。父进程在接受到该信号后,在信号处理函数中调用wait或者waitpid。
  3. 将僵尸进程的父进程杀掉,将最终使用的子进程变为孤儿进程,从而交由init进程处理其退出信息。

3.僵尸进程的危害

  1. 进程的退出状态必须被维持下去,因为他要告诉关心它的进程(父进程),你交给我的任务,我办的怎么样了。可父进程如果一直不读取,那子进程就一直处于Z状态。
  2. 维护退出状态本身就是要用数据维护,也属于进程基本信息,所以保存task_struct(PCB)中,换句话说, Z状态一直不退出, PCB一直都要维护。
  3. 那一个父进程创建了很多子进程,就是不回收,是不是就会造成内存资源的浪费?是的!因为数据结构对象本身就要占用内存,想想C中定义一个结构体变量(对象),是要在内存的某个位置进行开辟空间!
  4. 僵尸进程会导致内存泄漏。

4.僵尸进程的模拟实现

代码如下:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/types.h>
  5. #include <sys/wait.h>
  6. int main()
  7. {
  8. pid_t id = fork();
  9. if(id < 0)
  10. {
  11. perror("fork");
  12. return 1;
  13. }
  14. else if (id > 0)
  15. {
  16. printf("parent[%d] is sleeping...\n",getpid());
  17. sleep(10);
  18. }
  19. else
  20. {
  21. // child进程
  22. printf("child[%d] is begin Z...\n",getpid());
  23. sleep(2);
  24. exit(EXIT_SUCCESS);
  25. }
  26. return 0;
  27. }


二、孤儿进程 

1.孤儿进程的概念

父进程结束或者异常终止,但是子进程继续运行。此时子进程的PPID被设置为1,即init进程。init进程接管了该子进程,并等待它结束,在父进程退出之后,子进程退出之前,该子进程属于孤儿进程 

2.模拟实现孤儿进程

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. int main()
  5. {
  6. pid_t id = fork();
  7. if (id < 0)
  8. {
  9. // 创建失败
  10. perror("fork");
  11. return 1;
  12. }
  13. else if (id == 0)
  14. {
  15. // child
  16. while (1)
  17. {
  18. printf("I am child, pid : %d\n", getpid());
  19. sleep(1);
  20. }
  21. }
  22. else
  23. {
  24. // father
  25. printf("I am father, pid : %d\n", getpid());
  26. sleep(10);
  27. exit(0);
  28. }
  29. return 0;
  30. }

 

三、进程优先级

1.进程优先级基本概念

2.查看系统进程

使用 ps -al 命令

 3.PRI and NI

为啥Linux进程的优先级要有数值范围? 

  1. 操作系统要较为均匀的调度进程,每个进程都要得到调度。
  2. 如果一个进程的优先级太大(太小),就会导致一直占用cpu(一直得不到调度---->进程饥饿)

  • 需要强调一点的是,进程的nice值不是进程的优先级,他们不是一个概念,但是进程nice值会影响到进程的优先级变化。
  • 可以理解nice值是进程优先级的修正修正数据。

4.查看进程优先级的命令(top)

top命令更改已经存在进程的nice值

进入top后按“r”–>输入进程PID–>输入nice值。

5.为啥要有优先级 

 

四、环境变量

1.环境变量基本概念

  • 环境变量(environment variables)一般是指在操作系统中用来指定操作系统运行环境的一些参数。
  • 如:我们在编写C/C++代码的时候,在链接的时候,从来不知道我们的所链接的动态静态库在哪里,但是照样可以链接成功,生成可执行程序,原因就是有相关环境变量帮助编译器进行查找。
  • 环境变量通常具有某些特殊用途,还有在系统当中通常具有全局特性。


2.为什么需要环境变量

环境变量(envirnment variables)一般是指在操作系统中用来指定操作系统运行环境的一些参数。

3.常见的环境变量 

4.查看环境变量的方法 

5.测试PATH

创建hello.cpp文件

  1. #include<iostream>
  2. int main()
  3. {
  4. std::cout<<"Hello world"<<std::endl;
  5. return 0;
  6. }

编译后运行 

 

我们分别带路径和不带路径运行,发现带路径就可以运行,而不带路径无法运行。 

这个时候就有个问题?

为什么有些指令可以直接执行(例如ls、cd等等),不需要带路径,而我们的二进制程序需要带路径才能执行? 

如果我们的程序所在路径加入环境变量PATH当中会发生什么呢?

 

我们发现确实不用带路径就可以运行了 。这个时候我们又有问题了。

为啥这个时候有可以不带路径了,我只是把mybin的路径加入到环境变量中而已? 

6.环境变量的组织方式

我们在启动shell的时候就会自动执行的一个可执行程序


在C语言中我们学到了指针数组,就是存放指针的数组,那么存放的就是各种环境变量的首地址,通过找到环境变量的首地址就可以找到对于的环境变量。
在这里插入图片描述
每个程序都会收到一张环境表,环境表是一个字符指针数组,每个指针指向一个以’\0’结尾的环境字符串


7.和环境变量相关的命令

  • echo: 显示某个环境变量值。
  • export: 设置一个新的环境变量。
  • env: 显示所有环境变量。
  • unset: 清除环境变量。
  • set: 显示本地定义的shell变量和环境变量。

8.通过代码如何获取环境变量

1.命令行三个参数

  1. #include<stdio.h>
  2. int main(int argc,char* argv[],char* env[])
  3. {
  4. //argc : 命令行参数的个数, 本质上就是argv数组的元素个数
  5. //argv :具体的命令行参数
  6. // envp : 环境变量的值
  7. int i = 0;
  8. for(i=0;i<argc;i++)
  9. {
  10. printf("%s\n",argv[i]);
  11. }
  12. for(i=0;env[i]!=NULL;i++)
  13. {
  14. printf("%s\n",env[i]);
  15. }
  16. return 0;
  17. }

2.通过第三方变量environ获取

  1. #include<stdio.h>
  2. int main(int argc,char* argv[])
  3. {
  4. extern char** environ;
  5. for(int i=0;environ[i]!=NULL;i++)
  6. {
  7. printf("%s\n",environ[i]);
  8. }
  9. return 0;
  10. }

3.系统调用getenv和putenv函数

  1. #include<stdio.h>
  2. #include<unistd.h>
  3. #include<stdlib.h>
  4. int main()
  5. {
  6. printf("%s\n",getenv("PATH"));
  7. return 0;
  8. }


9.本地变量和环境变量以及内键命令

本地变量

 但是,有个问题我使用 echo $a ,为什么可以查看到本地变量 a 呢?

 我们再来实验一下,我们直接将环境变量搞挂掉。

 我们发现 ls、cat 等命令无法使用了,但是 echo 和 pwd 还是可以使用的,为啥呢?

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/煮酒与君饮/article/detail/991585
推荐阅读
相关标签
  

闽ICP备14008679号