【Linux系统编程】第二十弹---进程优先级 命令行参数 环境变量

news/2024/9/19 18:48:50/ 标签: linux, 服务器, 运维, C语言, 算法, 开发语言

个人主页: 熬夜学编程的小林

💗系列专栏: 【C语言详解】 【数据结构详解】【C++详解】【Linux系统编程】

目录

1、进程优先级

2.1、什么是优先级

2.2、优先级的描述

2.3、优先级与权限的关系

2.4、为什么要有优先级

2.5、Linux优先级的特点

2.6、其他概念

2、命令行参数

3、环境变量

3.1、基本概念

3.2、PATH环境变量

3.3、其他环境变量


1、进程优先级

2.1、什么是优先级

指定进程获取某种资源(CPU)的先后顺序。

2.2、优先级的描述

进程优先级的描述跟进程状态描述类似,实质是在task_struct(内存控制块)结构体内部有一个描述优先级的属性,通过数字表示先后顺序。Linux 中优先级数字越小,优先级越高。

2.3、优先级与权限的关系

权限决定能不能获取资源。

优先级决定获取资源的顺序,已经能获取资源。

2.4、为什么要有优先级

进程访问的资源(CPU)始终是有限的。系统中进程大部分情况都是比较多的。

操作系统关于调度和优先级的原则:分时操作系统,基于时间片进行调度,保证基本的公平。如果进程因为长时间不被调度,就造成了饥饿问题。

2.5、Linux优先级的特点

在讲解优先级特点之前,我们先通过一个C语言程序查看优先级。

C语言代码

#include<stdio.h>
#include<unistd.h>int main()
{while(1){printf("I am a process,pid:%d\n",getpid());}return 0;
}

 命令行代码

ps -al // 查看所有进程信息

测试结果 

我们很容易注意到其中的几个重要信息,有下:

UID : 代表执行者的身份。
PID : 代表这个进程的代号。
PPID :代表这个进程是由哪个进程发展衍生而来的,亦即父进程的代号。
PRI :代表这个进程可被执行的优先级,其值越小越早被执行。
NI :代表这个进程优先级的修正数据,nice值,新的优先级 = 默认优先级 + nice,达到对于进程优先级动态修改的过程。

调整优先级

nice/renice 调整优先级

此处博主使用top命令调整优先级。

  • 输入top命令
  • 进入top后按“r”–>输入进程PID–>输入nice值

测试一 

我们将nice值改为100,现象如下:

可以看到 NI 只修改为了19,PRI修改为了99。表名nice值并不能任意调整,而是有范围的。 范围是 [-20,19] ,共40个数字。

测试二

我们将nice值改为-10,现象如下:

我们可以看到,当我们第二次修改nice值时,不允许我们修改,因为修改优先级是有风险的,如果强制需要修改nice值,我们只需要切换成root账号即可。 

测试三

使用root账号将nice值改为-10,现象如下:

我们可以看到NI修改成了-10,PRI修改成了70,我们刚刚的PRI是99,将nice值修改为-10,为什么现在的PRI为70了呢?原因是 每次调整优先级都是从80开始的。新的优先级 = 默认优先级 + nice,70 = 80 - 10。

测试四

使用root账号将nice值改为-100,现象如下:

可以看到NI被修改为-20,因此能够证明nice的最小值为-20,且新的优先级 = 默认优先级 + nice,60 = 80 - 20。

2.6、其他概念

  • 竞争性: 系统进程数目众多,而CPU资源只有少量,甚至1个,所以进程之间是具有竞争属性的。为了高效完成任务,更合理竞争相关资源,便具有了优先级。
  • 独立性: 多进程运行,需要独享各种资源,多进程运行期间互不干扰。
  • 并行: 多个进程在多个CPU下分别,同时进行运行,这称之为并行。
  • 并发: 多个进程在一个CPU下采用进程切换的方式,在一段时间之内,让多个进程都得以推进,称之为并发。

2、命令行参数

C语言代码中,main函数也是函数,可以带参数?

答案是可以带参数也可以不带参数,下面就使用带参数的main函数演示。

代码演示

#include<stdio.h>// 命令行参数测试
int main(int argc,char* argv[])
{int i = 0;for(i=0;i<argc;i++){printf("argv[%d] = %s\n",i,argv[i]);}return 0;
}

 测试结果

通过上图现象我们可以看到argc是元素个数,argv是一个边长数组,猜测数组以NULL结尾。 

验证数组以NULL结尾

#include<stdio.h>int main(int argc,char* argv[])
{int i = 0;// argv[i]为假则循环结束for(i=0;argv[i];i++){printf("argv[%d] = %s\n",i,argv[i]);}return 0;
}

测试结果 

为什么要有命令行参数?

本质:命令行参数本质是交给我们程序的不同选项,用来定制不同的程序功能。命令中会携带很多选项。

命令行中启动的程序是谁干的?

命令行中启动的程序,都会变成进程,且都是bash的子进程,因此是bash(命令行解释器)干的。

代码演示一

#include<stdio.h>
#include<unistd.h>int g_val = 10000;
int main()
{printf("I am parent process,g_val:%d,pid:%d,ppid:%d\n",g_val,getpid(),getppid());sleep(5);pid_t id = fork();if(id == 0){while(1){printf("I am child process,g_val:%d,pid:%d,ppid:%d\n",g_val,getpid(),getppid());sleep(1);}}else{printf("I am parent process,g_val:%d,pid:%d,ppid:%d\n",g_val,getpid(),getppid());sleep(1);}return 0;
}

父进程的数据,默认能被子进程看到并访问。

代码演示二

#include<stdio.h>
#include<unistd.h>
#include<string.h>int g_val = 10000;
int main(int argc,char* argv[])
{printf("I am parent process,g_val:%d,pid:%d,ppid:%d\n",g_val,getpid(),getppid());if(argc != 2){printf("Usage: %s -[a,b,c,d]\n", argv[0]);return 1;}if(strcmp(argv[1], "-a") == 0){printf("this is function1\n");}else if(strcmp(argv[1], "-b") == 0){printf("this is function2\n");}else if(strcmp(argv[1], "-c") == 0){printf("this is function3\n");}else if(strcmp(argv[1], "-d") == 0){printf("this is function4\n");}else{printf("no this function!!\n");}return 0;
}

测试结果 

命令行中启动的程序,都会变成进程,且都是bash的子进程。启动程序,默认是输入给父进程bash(命令行解释器)的!!!

3、环境变量

3.1、基本概念

  • 环境变量(environment variables)一般是指在操作系统中用来指定操作系统运行环境的一些参数。
  • 如:我们在编写C/C++代码的时候,在链接的时候,从来不知道我们的所链接的动态静态库在哪里,但是照样可以链接成功,生成可执行程序,原因就是有相关环境变量帮助编译器进行查找。
  • 环境变量通常具有某些特殊用途,还有在系统当中通常具有全局特性。

3.2、PATH环境变量

基本认知

执行命令和运行自己写的程序是没有区别的,且Linux中70% 的命令是C语言写的。

为什么执行ls这些系统提供的命令可以不带路径,而我们自己写的C语言程序需要加./(带路径)呢?

Linux系统重存在一些全局的设置,表明告诉命令行解释器应该去哪个路径下去寻找可执行程序,当执行ls命令(可执行程序在/usr/bin目录中)时,会先去找ls对应的可执行程序,默认去PATH中找。

查看PATH内容 

 通过echo $PATH 获取PATH(环境变量)内容 ,类似*p。

补充:

  • 系统中很多的配置,在我们登录Linux系统的时候,已经被加载到了bash进程中(内存)。
  • bash在执行命令的时候,需要先找到路径(默认去PATH中找),因为未来要加载。

如果我们想像系统指令一样执行自己写的可执行程序,怎么做?

方式一(粗暴):直接将我们的可执行程序拷贝到PATH的其中一个路径(例如/usr/bin)中。

注意:拷贝内容到PATH环境中需要配root权限,此处使用sudo提权。

sudo cp myprocess /usr/bin  # 此处的/usr/bin也可以是PATH中的其他路径

 不建议使用这种方式,因此下面我们将该可执行程序在/usr/bin中的内容删除。

sudo rm /usr/bin/myprocess

方式二(温柔):把可执行程序路径增加到PATH环境变量中。

错误示范:

PATH=/home/jkl/path # 将可执行程序的当前目录赋值给PATH变量,会直接覆盖PATH

环境变量直接被我们新的路径覆盖了怎么办呢?

其实很简单,直接将系统关闭,重进系统就可以了。因为我们上面说了 系统中很多的配置,在我们登录Linux系统的时候,已经被加载到了bash进程中(内存)

我们重新登录系统之后,发现环境变量回来了,且不能直接执行我们写的可执行程序了。

 正确示范:

PATH=$PATH:/home/jkl/path # 将原本的PATH路径以及我们可执行程序当前目录赋值给PATH

 演示结果

PATH环境变量是Linux系统中搜索可执行程序的默认路径,也是which指令搜索路径的默认路径。

PATH环境变量的路径是内存级别的,重新登录系统又会变成默认路径,怎样让追加的环境变量路径永久存在呢?

最开始的环境变量不是在内存中,而是在系统对应的配置文件中,在我们登录系统时,会创建一个bash进程,bash进程会读取配置文件,然后把配置文件的环境变量(包括PATH)在自己的bash进程拷贝一份。

这个配置文件在哪里?

vim .bash_profile
vim .bashrc
vim /etc/bashrc

我们的PATH配置文件可能在.bash_profile或者.bashrc中。博主的在.bashrc中。

想要登录时每次都默认执行自己写的可执行程序,把可执行程序路径加到环境变量的配置文件里面即可,如下图:

加上该路径之后,myprocess可以像系统命令(ls)一样,在任意目录中使用了,且无需加./。 

Windows也有环境变量,我们在安装jdk或者python时一般需要配置环境变量。

3.3、其他环境变量

env : 查看系统所有的环境变量

history :查看历史命令

HOME : 家目录所在的环境变量

PWD : 当前目录路径变化

SHELL :  当前Shell,它的值通常是/bin/bash。

HISTSIZE : 历史命令个数 上翻下翻 Linux默认会记录最新的1000条命令

自己定义一个环境变量:

export THIS_IS_MY_ENV=hellolinux   导入环境变量(不加export,依旧存在变量,本地变量)

unset THIS_IS_MY_ENV  取消环境变量

export测试

疑问:export 也是一个命令,命令就会创建子进程,子进程就应该不被bash看到,为什么却能将变量导入到环境变量中。?

80% 命令都是bash创建子进程执行的 ,称为普通命令。还有20%命令,如(export / echo) 由bash亲自执行的,称为内建命令

怎么证明真的有内建命令呢?

通过上面实验,我们可以证明确实有内建命令。 

普通测试 

  • 本地变量只在bash内部有效,无法被子进程继承下去。导成环境变量才能被获取。
  • 用本地变量理解内建命令。内建命令可以查到本地变量。
  • echo能够打印本地变量(子进程无法获取)也能够证明echo是内建命令。

注意:已经存在的本地变量导成环境变量,只需要使用export 变量名即可。

能否通过C语言程序查看环境变量呢?

答案是可以的,在写程序之前我们需要查一下手册,man environ,environ为查看全局的指针变量。

C语言程序

#include<stdio.h>
#include<unistd.h>
int main()
{extern char** environ;// 声明外部文件变量int i=0;for(i=0;environ[i];i++)// environ[i]为假则循环结束{printf("env[%d]->%s\n",i,environ[i]);}return 0;
}

测试结果 

测试结果与使用命令env的结果是一样的。 说明环境变量默认也是可以被bash子进程拿到的。环境变量们,默认在bash内部。

环境变量有很多,bash内部是如何组织的?

通过上面的C语言代码我们大概可以推断,环境变量的组织与命令行参数类似,是通过一个变长数组组织的,最后一个数据以NULL结尾。

main函数带参的C语言程序

#include<stdio.h>
#include<unistd.h>
int main(int argc,char* argv[],char* env[])
{int i = 0;for(i=0;env[i];i++){printf("env[%d]->%s\n",i,env[i]);}return 0;
}

测试结果 

bash进程启动的时候,默认会给子进程形成argv[]命令行参数表(用户输入的命令行来),env[]环境变量表(从OS的配置文件来),bash通过各种方式交给子进程。

补充:

  • 导环境变量的本质:把字符串添加到环境变量表中。
  • 环境变量具有系统级的全局属性,因为环境变量会被子进程继承下去。

获取环境变量方法:

  • extern char** environ;
  • main函数参数
  • getenv("path");
man getenv

代码 

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
int main(int argc,char* argv[],char* env[])
{char* path = getenv("PATH");if(path == NULL) return 1;printf("psth:%s\n",path);return 0;
}

测试结果 

putenv(); // 创建环境变量


http://www.ppmy.cn/news/1527284.html

相关文章

Brave编译指南2024 Windows篇:拉取Brave源码(六)

1.引言 在编译Brave浏览器之前&#xff0c;首先需要获取项目的源码。源码是Brave浏览器的核心组成部分&#xff0c;包含了所有的功能、特性和组件代码。通过获取和管理源码&#xff0c;开发者可以深入了解Brave的实现细节&#xff0c;并进行二次开发和定制化改进。为了确保源码…

【Git】Clone

当git clone失败时&#xff0c;出现 RPC failed; curl 92 HTTP/2 stream 0 was not closed cleanly: CANCEL (err 8) 错误&#xff0c;可能由于网络连接不稳定或仓库太大导致的。 可以尝试以下几种方法来解决这个问题&#xff1a; 增加 Git 的缓冲区大小&#xff1a; git confi…

go语言的基本语法

学了go语言但是一直没整理。。。那怎么证明我学了&#xff1f;如果学了之后忘了怎么复习&#xff1f;遂诞生这几篇&#xff0c;当作Linux中间的小插曲 整理一下go语言的基本语法&#xff1a; package mainimport ("bufio""fmt""os" ) 在使用对…

蓝桥杯-STM32G431RBT6(串口)

前言 一、配置 二、使用步骤 1.串口发送 代码逻辑 效果展示 2.串口接收单个字符 代码逻辑 中断回调函数 3.串口接受字符串 代码逻辑 字符串函数 中断回调函数 声明 代码开源 前言 一、配置 二、使用步骤 1.串口发送 代码逻辑 sprintf(tx_buf,"jin ke\r\n&…

十大排序之:冒泡排序

目录 一、简介 实现过程 时间复杂度 二、代码实现 函数声明 Swap函数 单趟 多趟 测试 优化 一、简介 冒泡排序是一种简单的排序算法&#xff0c;它重复地比较相邻的两个元素&#xff0c;如果顺序错误就交换它们&#xff0c;直到没有元素需要交换为止。这个过程类…

通信工程学习:什么是HSS归属用户服务器

HSS&#xff1a;归属用户服务器 HSS&#xff08;归属用户服务器&#xff0c;Home Subscriber Server&#xff09;是IP多媒体子系统&#xff08;IMS&#xff09;中控制层的一个重要组成部分&#xff0c;它扮演着存储和管理用户相关信息的核心角色。以下是关于HSS归属用户服务器的…

反激电路中TL431光耦反馈参数的计算,环路设计思路

反馈的过程 当副边的输出电压升高时&#xff0c;TL431参考端电压&#xff08;R端&#xff09;电压也会升高&#xff0c;使得TL431的导通量增加&#xff0c;同时光耦内部的发光二极管流过的电流也会增大&#xff0c;进而使得光耦三极管导通量增加&#xff0c;相连的电源IC电压反…

【springboot】简易模块化开发项目整合MyBatis-plus

接父子工程项目搭建&#xff0c;继续扩展项目 重新调整模块结构 1.删除子模块fast-demo-web中所有无用内容 2.右键fast-demo-web名称->新建&#xff08;news&#xff09;->模块&#xff08;Module&#xff09;&#xff0c;创建新的子模块 3.选择新建Maven工程 4.新建模…

【STL】 set 与 multiset:基础、操作与应用

在 C 标准库中&#xff0c;set 和 multiset 是两个非常常见的关联容器&#xff0c;主要用于存储和管理具有一定规则的数据集合。本文将详细讲解如何使用这两个容器&#xff0c;并结合实例代码&#xff0c;分析其操作和特性。 0.基础操作概览 0.1.构造&#xff1a; set<T&…

浅谈WebService

一、基本介绍 WebService是一个术语&#xff0c;通常用来描述一种软件系统&#xff0c;它通过网络提供服务。这些服务可以是任何形式的&#xff0c;比如执行计算、处理数据、提供信息等。WebServices使得不同的应用程序能够通过标准化的协议&#xff08;如HTTP&#xff09;进行…

【Linux】多路转接epoll

一、I/O多路转接 poll 1.1 poll函数接口 函数原型 函数参数 fds&#xff1a;是一个poll函数监听的结构列表&#xff0c;每一个元素中包含了三部分内容&#xff1a;文件描述符&#xff0c;监听的事件集合&#xff0c;返回的事件集合。nfds&#xff1a;表示的是fds数组的长度tim…

7 种有助于压缩图像的最佳图像压缩工具

您的网站是否由于图像文件过大而表现不佳&#xff1f;现在是时候探索最适合您网站的图像压缩工具了。 影响页面加载速度的因素之一是需要更多资源来加载的庞大视觉对象&#xff0c;例如叠加图像。在这种情况下&#xff0c;您应该先压缩视觉对象&#xff0c;然后再将其上传到您…

QT程序的安装包制作教程

在Windows平台上开发完qt c桌面应用程序以后&#xff0c;需要制作一个安装包&#xff0c;方便生产和刻盘交货&#xff0c;本文记录相关流程。 目录 一、安装Qt Installer Framework 二、准备可执行程序 2.1 生成Release程序 2.2 完成依赖库拷贝 三、创建安装包程序 一、…

校园社团|基于springBoot的校园社团信息管理系统设计与实现(附项目源码+论文+数据库)

私信或留言即免费送开题报告和任务书&#xff08;可指定任意题目&#xff09; 目录 一、摘要 二、相关技术 三、系统设计 四、数据库设计 五、核心代码 六、论文参考 七、源码获取 一、摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信…

关于前后端跨域访问的问题

在vue项目的vue.config.js配置文件里面添加下面的内容 http://localhost:8080这是后端tomcat端口&#xff0c;将后端的路径的tomcat端口换成/api

python数据分析知识点大全

Python数据分析知识点大全可以归纳为以下几个主要方面&#xff1a; 一、基础概念与目的 数据分析定义&#xff1a;数据分析是指用适当的统计分析方法对收集来的大量数据进行分析&#xff0c;提取有用信息和形成结论&#xff0c;对数据加以详细研究和概括总结的过程。其目的在…

34.贪心算法1

0.贪心算法 1.柠檬水找零&#xff08;easy&#xff09; . - 力扣&#xff08;LeetCode&#xff09; 题目解析 算法原理 代码 class Solution {public boolean lemonadeChange(int[] bills) {int five 0, ten 0;for (int x : bills) {if (x 5) // 5 元&#xff1a;直接收下…

基于微信小程序的科创微应用平台设计与实现+ssm(lw+演示+源码+运行)

基于微信小程序的科创微应用平台 摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了基于微信小程序的科创微应用平台的开发全过程。通过分析基于微信小程序的科创微应用平台管理的不足&#xff0c;创建了一个计…

基于PSO-SVM的乳腺癌数据分类识别算法matlab仿真,对比BP神经网络和SVM

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 4.1 PSO粒子群优化 4.2 svm 4.3 PSO-SVM 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) pso优化SVM过程&#xff1a; 识别率对比&#xff1a; 2.算法运行软件版本 …

docker如何实现资源隔离

Docker 通过多种机制实现了资源隔离&#xff0c;这些机制包括命名空间&#xff08;namespaces&#xff09;、控制组&#xff08;control groups, cgroups&#xff09;以及其他容器相关的技术。下面详细介绍 Docker 如何使用这些技术来实现资源隔离。 1. 命名空间&#xff08;N…