【Linux】————进程控制

铁佛  金牌会员 | 2024-10-14 09:03:24 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 822|帖子 822|积分 2466

 

                                                      作者主页:     作者主页
                                                      本篇博客专栏:Linux专栏
                                                      创作时间 :2024年10月10日



一、程序地址空间:

1、C/C++中的程序地址空间:


在c++中我们了解了这样的空间分布图。
   我们应怎样去创建和访问变量呢?
  本质就是:起始地址+偏移量(实在我们的变量类型就是偏移量)
  

   但是上面这些实在不是内存!!!
  我们下面来做一个小实验!!!
  
  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. int g_val = 0;
  5. int main()
  6. {
  7.      pid_t id = fork();
  8.      if(id < 0)
  9.      {
  10.          perror("fork");
  11.          return 0;
  12.      }
  13.      else if(id == 0)   
  14.      {
  15.          //child
  16.          printf("child[%d]: %d : %p\n", getpid(), g_val, &g_val);
  17.      }
  18.      else
  19.      {     
  20.          //parent
  21.          printf("parent[%d]: %d : %p\n", getpid(), g_val, &g_val);
  22.      }
  23.      sleep(1);
  24.      return 0;
  25. }
复制代码

   这里我们可以看到上面这两个输出的变量和地址是雷同的,这就说明子进程是按照父进程的模板得到的,父进程没有将代码进行修改,但是只要将代码轻微改一下。
   
  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. int g_val = 0;
  5. int main()
  6. {
  7.      pid_t id = fork();
  8.      if(id < 0)
  9.      {
  10.          perror("fork");
  11.          return 0;
  12.      }
  13.      else if(id == 0)   
  14.      {
  15.          //child
  16.          g_val = 100;
  17.          printf("child[%d]: %d : %p\n", getpid(), g_val, &g_val);
  18.      }
  19.      else
  20.      {     
  21.          //parent
  22.          printf("parent[%d]: %d : %p\n", getpid(), g_val, &g_val);
  23.      }
  24.      sleep(1);
  25.      return 0;
  26. }
复制代码

   这里我们发现,两个进程的地址雷同,但是对应的变量的内容竟然差别,这是为什么呢?
    我们可以得出以下的结论:
  变量的内容不一样,父子进程绝对不是输出的同一个变量
  但地址值是一样的,说明这个地址绝对不是物理地址
  在Linux下,这种地址叫做假造地址
  我们在C/C++中看到的地址,全部都是假造地址,用户是看不到物理地址的,用OS同一管理
  OS负责将假造地址转换为物理地址
  2、进程地址空间:


   上面的图就可以说明问题,同一个变量,地址雷同,实在是假造地址雷同,内容差别实在是被映射到了差别的物理地址。 
    那么我们应该怎样去理解假造地址雷同而物理地址差别的问题呢?
  
  

  • 父进程有自己的假造地址,也有自己的页表
  • 子进程被创建时,父进程会将自己的页表也给到子进程
  • 但是子进程在改变数据时,会发生写时拷贝,物理地址改变了,但是假造地址没有改变
  


什么是地址空间?什么是区域划分?

我们在创建进程的时候不仅要有 pcb,也要管理地址空间(先形貌,在组织),有一个 struct mm_struct 的结构体。


为啥要有地址空间?

   1、让进程以同一的视角对待内存,通过假造地址加页表,可以将乱序的内存变为有序,分门别类的规划好,乱序---->有序
  2、存在假造地址空间,可以有效地进行进程访问内存的安全查抄
  我们怎样去理解 存在假造地址空间,可以有效的进行进程内存的安全查抄呢?
我提一个问题,我们 常量区的变量 为啥不能修改呢?
我们页表中除了有映射外,另有权限的限制,当进程要修改常量区的变量时,直接在页表就没有权限。

 地址空间的增补

   每个进程都有自己的页表。
  
二、进程创建:

1.作者主页

   在linux中fork函数时非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。
  

   进程调用fork,当控制转移到内核中的fork代码后,内核做:
  

   当一个进程调用fork之后,就有两个二进制代码雷同的进程。而且它们都运行到雷同的地方。但每个进程都将可以开始它们自己的路程,看如下程序: 
   
  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. int main()
  5. {
  6.      printf("Before: %d\n", getpid());
  7.      pid_t id = fork();
  8.      if(id < 0)
  9.      {
  10.          perror("fork");
  11.          return 0;
  12.      }
  13.      else if(id == 0)   
  14.      {
  15.          //child
  16.          printf("child: %d \n", getpid());
  17.          sleep(2);
  18.      }
  19.      else
  20.      {     
  21.          //parent
  22.          printf("parent: %d \n", getpid());
  23.          sleep(2);
  24.      }
  25.      
  26.      return 0;
  27. }
复制代码

   从效果中可看出,fork之前父进程独立执行,之后父子进程分别执行,执行先后由调理器完全决定
    此中,默认环境下,父子进程共享代码,但是数据各有一份(但是如果父子进程只对数据进行读取,不须要私有)
  程序=代码(逻辑)+数据
  代码共享:所有进程共享代码,不外一般都是fork执行之后,为啥代码是共享的,由于代码不可以修改,所以是共享的
  为啥各自的数据要私有一份呢,由于进程之间具有独立性,数据是很多的,且不是所有的数据都要全部拷贝,把本来可以在后面拷贝,甚至不要拷贝的数据,都拷贝了,就比力浪费时间和空间,所以拷贝的过程不是立马做的,而是写时拷贝!!!
  2、写时拷贝


通常,父子代码共享,父子再不写入时,数据也是共享的,当恣意一方试图写入,便以写时拷贝的方式各自一份副本。具体见下图:

   父子进程代码共享,数据独有:当恣意一方试图写入,便以写时拷贝的方式拷贝一份副本
  3.fork通例用法

   

  • 一个父进程希望复制自己,使父子进程同时执行差别的代码段。比方,父进程等待客户端哀求,天生子进程来处置处罚哀求。
  • 一个进程要执行一个差别的程序。比方子进程从fork返回后,调用exec函数。
  4.fork调用失败的原因

   

  • 系统中有太多的进程。
  • 现实用户的进程数超过了限制。
  三、进程终止

1.进程终止的概念


   main 函数的返回值可以被父进程获取的,用来判定子进程的干活的环境 。
  

   检察上一个进程的退出码 
  echo $?
  

   我们父进程就可以通过这两个数字来判定子进程的退出环境 。
  代码非常终止,退出码就没有意义了!!!
   2.进程常见退出方法



3._exit函数 

   

  • _exit函数 是系统调用函数。
  • _exit函数 在终止进程的时候,不会主动革新缓冲区。
  


4.exit函数

   

  • exit函数 是库函数。
  • exit函数 在终止进程的时候,会主动革新缓冲区。
  

   exit最后也会调用_exit, 但在调用_exit之前,还做了其他工作:
  

  • 执行用户通过 atexit或on_exit定义的清理函数。
  • 关闭所有打开的流,所有的缓存数据均被写入。
  • 调用_exit。
  


四、进程等待

1.什么是进程等待

   通过 wait/waitpid 的方式,让父进程(一般环境)对子进程进行资源回收等待过程!!!
   
之前讲过,子进程退出,父进程如果不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存走漏。
另外,进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,由于谁也没有办法杀死一个已经死去的进程。
最后,父进程派给子进程的任务完成的怎样,我们须要知道。如,子进程运行完成,效果对还是不对,或者是否正常退出。
父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息。
  2.进程程序更换原理 


   用fork创建子进程后执行的是和父进程雷同的程序(但有可能执行差别的代码分支),子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序更换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后该进程的id并未改变。
    我们就可以表明 进程程序更换后另有一个printf没有执行,为啥呢?
  

 3.进程程序更换的函数






最后:

非常感谢你可以耐着性子把它读完和我可以坚持写到这里,送几句话,对你,也对我:
1.一个冷知识:
屏蔽力是一个人最顶级的本领,任何斲丧你的人和事,多看一眼都是你的不对。

2.你不用变得很外向,内向挺好的,但须要你发言的时候,一定要大胆。
正所谓:君子可内敛不可脆弱,面不公可起而论之。

3.成年人的世界,只筛选,不教育。
4.自律不是6点起床,7点准时学习,而是不管别人怎么说怎么看,你也会坚持去做,绝不打乱自己的节奏,是一种自我的恒心。
5.你开始炫耀自己,往往都是劫难的开始,就像老子在《道德经》里写到:光而不耀,静水流深。
最后如果以为我写的还不错,请不要忘记点赞✌,收藏✌,加关注✌哦(。・ω・。)
愿我们一起加油,奔向更优美的未来,愿我们从懵懵懂懂的一枚菜鸟渐渐成为大佬。加油,为自己点赞!


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

铁佛

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表