分享知乎上看到的一句话,共勉: 学习周期分为学习,思考,实践,校正四个阶段,周期越短,学习效率越高。

      前面讲的都是操作系统如何管理进程,接下来,看看用户如何进行进程控制。

    1.进程创建

      先介绍一下函数: pid_t fork(void);   返回值: 子进程返回0 父进程返回子进程id,创建子进程失败则返回-1       

        当一个进程调用fork后,就会产生两个二进制代码相同的代码。并且它们都运行到相同的地方。现在,给大家演示一下:

     

      

     由此,我们得到验证: fork之前父进程独立运行,fork之后,父子两个执行流分别执行。

    fork原理:       

          进程调用fork后,当控制权转移到内核的时候,内核会做四件事:

          1.分配新的内存块和数据结构给子进程

          2.将父进程部分数据结构拷贝给子进程

          3.添加子进程到系统进程(PCB)列表中

               4.fork返回,开始调度

      写时拷贝技术: 

      

      实际上fork就是给子进程创建pcb,将父进程pcb数据拷贝过来,这时候父子进程因为虚拟地址空间页表完全一样,因此他们的代码和数据看起来都是一样的。

       但是,进程应该具有独立性,修改任意进程数据不应该影响其他进程,所以子进程也应该开辟自己的物理内存来存储数据。

       但是,如果子进程不修改数据,再开辟空间,更新页表,就浪费了资源,所以,操作系统一开始并不开辟内存拷贝数据,而是等一个进程修改数据后再给子进程开辟新的物理内存,拷贝数据---写时拷贝技术

      再简单介绍一下vfork --- 父子进程共用一块虚拟地址空间 , 因为vfork在很多系统上有问题, 所以不建议使用。

      vfork子进程运行,父进程阻塞,直到子进程结束。

      2.进程终止

    进程退出的时候,会保存退出原因,从而知道进程任务是否正确完成。

    进程退出场景:

          1.正常退出,结果正确;

          2.正常退出,结果错误;

            3.异常退出,结果不能作为判断标准

      进程的退出方式:

          1.main中return;

          2.调用exit函数;  --- 1,2都是先刷新缓冲区,做了其他释放工作才推出

          3.调用_exit函数;   --- 粗暴退出,直接释放所有资源

    

    3.进程等待

      为什么要进程等待呢? 因为要避免僵尸进程

      因为父进程不知道子进程什么时候退出,所以创建子进程后要一直等待子进程退出

      函数介绍:

        

         阻塞: 为了完成操作发起调用,如果当前不具备完成条件,则一直等待,直到完成操作。

         阻塞和非阻塞的区别: 发起调用后是否立即返回。 

        获取子进程status: 

          wait和waitpid都有一个输出型参数status,由操作系统填充。

          status不能简单当做整型来看,而是一个位图,具体细节如下:

                       

            测试验证代码:

            coredump标志 : 核心转储标志

            在程序异常退出时,保存程序的堆栈信息,方便事后调试  (核心转储通常默认是关闭的: 安全隐患 和 空间占用)

  

    4.程序替换

    前面我们知道,一个进程运行什么代码,取决于虚拟地址空间中的代码段映射物理地址中的那个真实代码区域。

      意味着如果将虚拟地址代码段映射到物理内存的代码位置替换成另一个程序的位置,那么进程将运行另一个程序。

      

      是什么 : 替换代码映射的代码位置成为另一个内存区域代码的位置,并且重新初始化数据段

       为什么 : 大多数情况下,我们创建一个子进程都是为了让子进程运行另一个程序,做其他的任务

       实现 : 操作系统提供了一套接口都能实现程序替换,统称为exec函数族     

#include<unistd.h>

int execl(const char* path,const char* arg,...);  //后面以NULL结束
int execlp(const char* file,const char* arg,...);
int execle(const char* path,const char* grg,...,char* const envp[]); int execv(const char* path,const char* argv); //argv数组也要以NULL结束
int execvp(const char* file,const char* argv);
int execve(const char* path,const char* argv,char* const envp[]); //envp数组也是NULL结束 //事实上,只有execve是系统调用,其它函数都调用了execve

      2.函数参数

      path:表示对应的文件在哪个目录下(也表示我要执行的程序是哪一个);

      file:只用填文件名,系统会在PATH下找;

      envp[]:表示自己设置的,替换后的程序的环境变量

      3.命令理解

      l(list):表示参数采用列表形式

      v(vector):参数采用数组形式

      p(path):有p自动搜索环境变量PATH

      e(envp):表示自己设置(维护)环境变量(自己设置的环境变量是什么,则最新程序的环境变量就是什么)

      理论讲了这么多,不如写个测试代码,看看怎么用吧:

#include <unistd.h>
int main()
{
char *const argv[] = {"ps", "-ef", NULL};
char *const envp[] = {"PATH=/bin:/usr/bin", "TERM=console", NULL};

execl("/bin/ps", "ps", "-ef", NULL);

// 带p的,可以使用环境变量PATH,无需写全路径
execlp("ps", "ps", "-ef", NULL);

// 带e的,需要自己组装环境变量
execle("ps", "ps", "-ef", NULL, envp);

execv("/bin/ps", argv);
// 带p的,可以使用环境变量PATH,无需写全路径
execvp("ps", argv);

// 带e的,需要自己组装环境变量
execve("/bin/ps", argv, envp);
exit(0);
}

  

Linux 进程控制的更多相关文章

  1. Linux进程控制(二)

    1. 进程的创建 Linux下有四类创建子进程的函数:system(),fork(),exec*(),popen() 1.1. system函数 原型: #include <stdlib.h&g ...

  2. Linux进程控制(一)

    1. Linux进程概述 进程是一个程序一次执行的过程,它和程序有本质区别.程序是静态的,它是一些保存在磁盘上的指令的有序集合:而进程是一个动态的概念,它是一个运行着的程序,包含了进程的动态创建.调度 ...

  3. Linux - 进程控制 代码(C)

    进程控制 代码(C) 本文地址:http://blog.csdn.net/caroline_wendy 输出进程ID.getpid(). 代码: /*By C.L.Wang * Eclipse CDT ...

  4. Linux进程控制(三)

    1. 进程间打开文件的继承 1.1. 用fork继承打开的文件 fork以后的子进程自动继承了父进程的打开的文件,继承以后,父进程关闭打开的文件不会对子进程造成影响. 示例: #include < ...

  5. Linux进程控制——exec函数族

    原文:http://www.cnblogs.com/hnrainll/archive/2011/07/23/2114854.html 1.简介 在Linux中,并不存在exec()函数,exec指的是 ...

  6. LINUX进程控制

    1. 引言 一个程序是存储在文件中的机器指令序列.一般它是由编译器将源代码编译成二进制格式的代码.运行一个程序意味着将这个机器指令序列载入内存然后让处理器(cpu)逐条执行这些指令. 在Unix术语中 ...

  7. linux进程控制命令

    & 加在一个命令的最后,可以把这个命令放到后台执行 ,如gftp &. ctrl + z 可以将一个正在前台执行的命令放到后台,并且处于暂停状态,不可执行. jobs 查看当前有多少在 ...

  8. linux 进程控制笔记

    进程创建 普通函数调用完成后,最多返回(return)一次,但fork/vfork会返回二次,一次返回给父进程,一次返回给子进程 父进程的返回值为子进程的进程ID,子进程的返回值为0 1.pid_t ...

  9. linux进程控制开发实例

    fork.c #include <sys/types.h> #include <unistd.h> #include <stdio.h> #include < ...

随机推荐

  1. Git是怎么Ignore文件的?

    Git is one of the most popular version control systems (VCS) available, especially thanks to hosting ...

  2. openwrt共享打印机需要安装哪几个文件

    opkg updateopkg install luci-app-p910ndopkg install kmod-usb-printer

  3. .net Core 中DateTime在Linux Docker中与Windows时间不一致

    最近写了一个.net core项目,部署到CentOS并在docker上运行的时候,发现DateTime.Now获取的时间与Windows不一致(定时执行的任务,晚了8个小时),在Windows中可以 ...

  4. [English]常用中英文对照表

    Always have been 一直如此 accordingly:相应地 assumption:假定 brace:大括号 branket:中括号 comma:逗号MISC:Miscellaneous ...

  5. Kubernetes addons 之 coredns部署

    Kubernetes addons 之 coredns部署 2019.06.04 18:04:35字数 1045阅读 121 DNS 是 Kubernetes 的核心功能之一,通过 kube-dns ...

  6. Visual Studio 2019更新到16.1.4

    Visual Studio 2019更新到16.1.4 此次更新仍然是修复bug为主,涉及到Xcode 11无法启动模拟器.Xamarin.Android SDK 9.3.0.23的调试bug.

  7. Windows下通过命令行 获取文件详细信息

    @echo off echo version: wmic datafile where Name="C:\\Product\\File\\Release\\1.1.1\\File.exe&q ...

  8. typescript类 继承 修饰符

    //1.ts中类的定义 /* es5: function Person(name){ this.name=name; this.run=function(){ console.log(this.nam ...

  9. Python3基础 只有int类型,没有long类型

             Python : 3.7.3          OS : Ubuntu 18.04.2 LTS         IDE : pycharm-community-2019.1.3    ...

  10. shiro解决一个账号异地登录的问题

    如下,找到session中的信息删除即可,按照这个方式试了下.基本可用 在多台服务器部署时,前提必须实现session共享. /** * 登录认证 * @param token * @return * ...