Linux系统编程——进程控制

本文详细介绍了Linux系统编程中的进程控制,包括CPU与MMU的工作原理、进程控制块PCB、环境变量及其相关函数、进程创建与回收、exec函数族的使用,以及孤儿进程和僵尸进程的处理。通过学习,读者可以深入了解Linux进程管理的核心概念和技术。
摘要由CSDN通过智能技术生成

1. 进程相关

1.1 CPU 与 MMU

1.1.1 CPU

CPU
CPU 执行一条指令过程:预取器先从cache或者内存中取出一条指令,然后交给译码器分析,译码器分析该条指令需要用到哪个寄存器,并把相关数据存储到对应的寄存器中,ALU对其进行运算,然后把数据回写到寄存器中,最后再把数据放到缓冲区,然后由内存把数据传输到总线,再显示到设备上

1.1.2 MMU(内存管理单元)

MMU
MMU功能

  • 虚拟内存与物理内存的映射
  • 设置修改内存访问级别

虚拟地址:可用地址空间0-4G
假如虚拟地址使用了2KB,那么映射到物理内存大小应该为4KB,因为一个page作为物理内存的最小单位,大小为4KB。

1.3 进程控制块PCB

每个进程在内核中都有一个进程控制块(PCB)来维护进程相关的信息,Linux内核的进程控制块是task_struct结构体。
/usr/src/linux-headers-3.16.0-30(可能不一样)/include/linux/sched.h文件中可以查看struct task_struct 结构体定义。

其内部成员有很多,重点掌握以下部分即可:

  • 进程id。系统中每个进程有唯一的id,在C语言中用pid_t类型表示,其实就是一个非负整数。
  • 进程的状态,有就绪、运行、挂起、停止等状态。
  • 进程切换时需要保存和恢复的一些CPU寄存器。
  • 描述虚拟地址空间的信息。
  • 描述控制终端的信息。
  • 当前工作目录(Current Working Directory)。
  • umask掩码。
  • 文件描述符表,包含很多指向file结构体的指针。
  • 和信号相关的信息。
  • 用户id和组id。
  • 会话(Session)和进程组。
  • 进程可以使用的资源上限(Resource Limit)。ulimit -a

2. 环境变量

环境变量,是指在操作系统中用来指定操作系统运行环境的一些参数

通常具备以下特征

  • ① 字符串(本质)
  • ② 有统一的格式:名=值[:值]
  • ③ 值用来描述进程环境信息。

存储形式:与命令行参数类似。char *[]数组,数组名environ,内部存储字符串,NULL作为哨兵结尾。
使用形式:与命令行参数类似。
加载位置:与命令行参数类似。位于用户区,高于stack的起始位置。
引入环境变量表:须声明环境变量。extern char ** environ;

2.1 常见环境变量

按照惯例,环境变量字符串都是name=value这样的形式,大多数name由大写字母加下划线组成,一般把name的部分叫做环境变量,value的部分则是环境变量的值。环境变量定义了进程的运行环境,一些比较重要的环境变量的含义如下:
PATH
可执行文件的搜索路径。ls命令也是一个程序,执行它不需要提供完整的路径名/bin/ls,然而通常我们执行当前目录下的程序a.out却需要提供完整的路径名./a.out,这是因为PATH环境变量的值里面包含了ls命令所在的目录/bin,却不包含a.out所在的目录。PATH环境变量的值可以包含多个目录,用:号隔开。在Shell中用echo命令可以查看这个环境变量的值:echo $PATH

SHELL
当前Shell,它的值通常是/bin/bash。echo $SHELL
TERM
当前终端类型,在图形界面终端下它的值通常是xterm,终端类型决定了一些程序的输出显示方式,比如图形界面终端可以显示汉字,而字符终端一般不行。
LANG
语言和locale,决定了字符编码以及时间、货币等信息的显示格式。
HOME
当前用户主目录的路径,很多程序需要在主目录下保存配置文件,使得每个用户在运行该程序时都有自己的一套配置。

练习:打印当前进程的所有环境变量。

#include <stdio.h>
#include <iostream>
using namespace std;
extern char **environ;//须声明环境变量

int main(){
   
    for(int i = 0;environ[i];i++){
   
        cout << environ[i] << endl;
    }
    return 0;
}

2.2 相关环境变量函数

2.2.1 getenv函数

  • 获取环境变量值
  • char *getenv(const char *name);
    • 成功:返回环境变量的值;
    • 失败:NULL (name不存在)

2.2.2 setenv函数

  • 设置或添加环境变量的值
  • int setenv(const char *name, const char *value, int overwrite);
    • 成功:0;失败:-1
    • 参数overwrite取值:
      • 1:覆盖原环境变量
      • 0:不覆盖。(该参数常用于设置新环境变量,如:ABC = haha-day-night)

2.2.1 unsetenv函数

  • 删除环境变量name的定义
  • int unsetenv(const char *name);
    • 成功:0;失败:-1
  • 注意事项:name不存在仍返回0(成功),当name命名为"ABC="时则会出错。
#include <iostream>
#include <stdlib.h>
#include <string.h>
using namespace std;

int main(){
   
    
    const char* name  = "ABD";
    char *val;
    val = getenv(name);//获取环境变量ABD的值,此时环境变量不存在,所以为空
    if(val){
   
        cout << name << ": " << val << endl;
    }else{
   
        cout << "环境变量不存在" << endl;
    }

    setenv(name,"day-day-up",1);//设置环境变量的值为day-day-up
    val = getenv(name);
    cout << name << ": " << val << endl;

    int ret  = unsetenv("ABCD");//删除环境变量的定义
    cout << "环境变量ABCD 的值为:" << ret << endl;

    ret = unsetenv("ABD");
    cout << "环境变量ABD的值为:" << ret << endl;
    cout << getenv(name) << endl;
    return 0;
}

3. 进程控制

3.1 进程ID相关函数

函数 函数原型 说明
getpid函数 pid_t getpid(void); 获取当前进程ID
getppid函数 pid_t getppid(void); 获取当前进程的父进程ID
getuid函数 uid_t getuid(void); 获取当前进程实际用户ID
geteuid()函数 uid_t geteuid(void); 获取当前进程有效用户ID
getgid函数 gid_t getgid(void); 获取当前进程使用用户组ID
getegid函数 gid_t getegid(void); 获取当前进程有效用户组ID

区分一个函数是“系统函数”还是“库函数”依据:

  • 是否访问内核数据结构
  • 是否访问外部硬件资源 二者有任一 → 系统函数;二者均无 → 库函数

3.2 子

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值