嵌入式技术
一、进程优先级
1、基本概念
cpu 资源分配的先后顺序,就是指进程的优先权(priority) .优先权高的进程有优先执行权利。配置进程优先权对多任务环境的 Linux 很有用,可以改善系统性能。 还可以把进程运行到指定的 CPU 上,这样一来,把不重要的进程安排到某个 CPU,可以大大改善系统整体性能。
2、查看以及修改系统进程的优先级
在 Linux 或者 unix 系统中,用ps –al命令则会类似输出以下几个内容, 其中:
UID : 代表执行者的身份
PID : 代表这个进程的代号
PPID :代表这个进程是由哪个进程发展衍生而来的,亦即父进程的代号
PRI :代表这个进程可被执行的优先级,其值越小越早被执行
NI :代表这个进程的 nice 值,其表示进程可被执行的优先级的修正数值。
我们看到我们现在有两个进程 bash ps 它们的进程优先级都是 80,如果我们要修改它们的优先级就要用到 NI 的 nice 值了。
P R I ( n e w ) = P R I ( o l d ) + n i c e PRI(new)=PRI(old)+nicePRI(new)=PRI(old)+nice
从这个公式中我们知道 新的 PRI = 老的 PRI + nice 值,但是这个老的 PRI 的值是指 PRI 最初的默认值,例如上面的bash ps 是 80,那么这个 PRI 以后不论怎么改老的 PRI 都是 80,当然大多数进程默认的 PRI 都是 80。
此外 nice 是有范围的!其取值范围是 - 20 至 19,一共 40 个级别。
于是下面我们去尝试去更改processC进程的优先级。
注意:将进程优先级调高(即将 nice 值设置为负数)需要 root 用户进行操作!
修改进程优先级的 Linux 指令
top命令
进入top后按 "r" -> 输入进程 PID -> 输入 nice 值
按下 “r”
输入 -20
再次[查看进程]的相关信息:
我们发现进程的优先级确实改变了,但是我们能改变进程优先级有限[ − 20 , 19 ] [-20,19][−20,19],因为调度器不允许我们将一个进程设置的优先级太高,进而导致其他进程难以被调度。
3、一些其他的关于进程优先级的指令和函数调用
nice指令,nice 命令的功能是用于调整进程的优先级,合理分配系统资源。-n 参数是 nice 值的优先级别,
以 nice 值为 -5 的方式执行指定程序
查看进程优先级
renice命令可以修改正在运行的进程的调度优先级。
renice更改一个或多个进程的调度优先级。第一个参数是要使用的优先级值,另一个参数被表示为进程标识信息。
renice [-n] priority [-gpu] identifier
-g, 后面加组的 pgid, 改变一个组的进程优先级
-u, 后面加 user name 或 uid,改变一个用户所拥有的进程优先级。
-p, 后面加 pid ,改变一个进程的进程优先级。
使用renice命令
函数调用
在 Linux 中关于改变进程优先级函数调用主要有两个:getpriority() 与setpriority()。
4、与进程优先级有关的一些进程性质
竞争性: 系统进程数目众多,而 CPU 资源只有少量,甚至 1 个,所以进程之间是具有竞争属性的。为了高效完成任务,更合理竞争相关资源,便具有了优先级
独立性: 多进程运行,需要独享各种资源,多进程运行期间互不干扰。
并行: 多个进程在多个 CPU 下,分别同时进行运行,这称之为并行
并发: 多个进程在一个 CPU 下采用进程切换的方式,在一段时间之内,让多个进程都得以推进,称之为并发
二、环境变量
1、基本概念
环境变量 (environment variables) 一般是指在操作系统中用来指定操作系统运行环境的一些参数,环境变量通常具有某些特殊用途,在系统当中通常具有全局特性。
如:我们在编写 C/C++ 代码的时候,在链接的时候,从来不知道我们的所链接的动态[静态库]在哪里,但是照样可以链接成功,生成可执行程序,原因就是有相关环境变量帮助编译器进行查找。
2、和环境变量相关的命令
1.env: 显示所有环境变量
2.echo: 显示的变量值 (需要带上 $ 符号)
3.export: 设置一个新的环境变量,或者将本地[变量提升]成环境变量。
4.unset: 清除环境变量
5.set: 显示本地定义的 shell 变量和环境变量
3、Linux 中的常见环境变量介绍
PATH : 指定命令的搜索路径
例如我们使用的 Linux 中 ls pwd命令,其实就是一个个 C 语言写的一个个小程序,为什么我们运行自己写的程序就要用./ + 自己的程序名 ,而我们运行ls pwd 从来不加./,这就和环境变量 PATH 有关了!
我们查看环境变量可以使用echo $环境变量命令:
默认情况下我们使用的 Linux 指令会去 PATH 路径下寻找源程序,由于ls指令的路径位置就在 PATH 的环境变量中所以我们可以不用加./
我们现在尝试将我们的路径添加到 PATH 环境变量里面,来让我们的程序也不需要加./,这时我们就需要使用一个新的指令了:export
export令可以将本地变量提升成环境变量,于是我们将我们的路径添加到 PATH 中就可以这样写:
export PATH=$PATH:你要添加的路径
当然我们使用export命令是暂时将本地变量提升为环境变量,当我们退出云服务器或关机重启都会消除export暂时提升的环境变量,想要真正的修改我们要修改相应的配置文件。
当然我们还可以将我们写的程序拷贝到 Linux 的 PATH 默认路径下,这样我们也不用使用./了,在 Linux 中,把可执行程序,拷贝到系统默认路径下,让我们可以直接访问的方式,相当于 Linux 下软件的安装!
HOME : 指定用户的主工作目录 (即用户登陆到 Linux 系统中时, 默认的目录)
由于 HOME 环境变量的存在,我们使用相同的命令cd ~却得到了不同的结果。
SHELL : 当前 Shell, 它的值通常是 / bin/bash。
4、环境变量的组织方式以及在 C 代码中如何获取环境变量
在 Shell 内部,环境变量其实是以环境变量表的方式进行维护的!
此外环境变量还具有全局性,我们知道我们在 bash 下运行的程序其父进程都是 bash, 那么 bash 就可以将自己的环境变量传递给子进程,并在子进程中发挥作用!
我们来看一段代码来验证环境变量具有全局性。
1. C 库函数getenv()获得单个环境变量
在看验证代码之前我们先了解一个函数getenv() getenv()是一个 C 库函数, 它可以获取一个环境变量的内容
函数原型:
函数的参数是环境变量的名称,返回值是一个char*字符串记录了环境变量里面的内容, 如果调用失败会返回 NULL 指针。
实例代码
#include#include int main() { char* env = getenv("USER");//USER是环境变量 if(env == NULL) { perror("getenv fail:"); } printf("%s ",env); return 0; }
代码输出结果
我们在代码里面多出来的USER变量就是来自 Shell 传递给我们的test1c进程的环境变量!
2. main()函数参数获得环境变量
此外我们我们还可以用main函数的参数来获得所有环境变量的地址,通过地址我们也能遍历所有环境变量
函数原型
int main(int argc, char *argv[], char *envp[]);
在这里我们先不谈论函数的参数 argc *argv[], 我们来谈论第三个参数!其中*envp[]是一个字符数组指针,指向的是一个指针数组,数组名代表首元素的地址,首元素是一个字符指针,*envp[]刚好又是指向首元素的指针,故其实*envp[]其实是一个二级指针!
明白了这些,我们来看下面一段代码:
#includeint main(int argc, int *argv[], int *envp[]) { for(int i =0; envp[i] != NULL ; ++i) { //打印所有环境变量,相当于 env 命令! printf("envp[%d]-->%s ", i, envp[i]); } return 0; }
可以看到我们确实打印出了所有的环境变量,而且这个环境变量来自于其父进程 bash。
3. C 语言全局变量environ获得环境变量
变量详情:
environ变量是一个二级指针与main()函数参数的char *envp[]类似。遍历所有环境变量也可以这样写:
#include#include int main() { extern char** environ; for(int i =0; environ[i] != NULL; ++i) { printf("environ[%d]-->%s ", i, environ[i]); } return 0; }
5、不同用户的环境变量是怎么形成的
通过上面的讲解我们知道了环境变量的概念与作用,环境变量中的每一个,都有自己的用途: 有的是进行路径查找的,有的时进行身份认证的,有的时进行动态库查找的,有的是用来进行确认当前路径等等每一个环境变量都有自己的特定应用场景。
我们也知道为什么,对于不同的用户其环境变量也并不相同,例如我们上面的 root 用户的环境变量与 pan 的环境变量有的一样有的不一样,那么 Linux 是怎样形成不同的环境变量的呢?
这里先给出结论:环境变量本质就是一个内存级的一张表,这张表由用户在登陆会统的时候,给特定用户形成属于自己的环境变量表。
在我们的家目录下有两个文件叫 .bashrc .bash_profile 在根目录下有一个bashrc的文件
打开这些文件看看!
6、main() 函数的命令行参数
在前面我们谈论中我们说到过main()函数的参数问题,我们还有两个参数没有谈论int argc char *argv[]。现在我们来讨论它们!
由于 C 语言中无法传递整个数组,所以在函数中想要获得数组元素的个数必须在传参时就要提前传递好,于是其中int argc 就是char *argv[]数组指针指向的数组的有效元素个数,不包含 NULL。
这个char *argv[]是一个数组指针,其指向的数组里面存放的都是char *的指针,这些char *的指针指向的内容需要我们使用命令行的方式进行设置。
我们先看下面一段代码:
#includeint main(int argc, int *argv[]) { printf("argc = %d ",argc); for(int i = 0; argv[i] != NULL; ++i) { printf("argv[%d]-->%s ", i, argv[i]); } return 0; }
运行结果:
我们 Linux 中ls命令有许多参数如-a -l -d -n,ls本质上就是 C 语言写的一个程序,它为什么能根据不同的参数执行不同的功能就是因为使用了 mian() 函数的命令行参数!
审核编辑:汤梓红
全部0条评论
快来发表一下你的评论吧 !