1. 进程相关
1.1 CPU 与 MMU
1.1.1 CPU
CPU 执行一条指令过程:预取器先从cache或者内存中取出一条指令,然后交给译码器分析,译码器分析该条指令需要用到哪个寄存器,并把相关数据存储到对应的寄存器中,ALU对其进行运算,然后把数据回写到寄存器中,最后再把数据放到缓冲区,然后由内存把数据传输到总线,再显示到设备上
1.1.2 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 |
区分一个函数是“系统函数”还是“库函数”依据:
- 是否访问内核数据结构
- 是否访问外部硬件资源 二者有任一 → 系统函数;二者均无 → 库函数