【Linux】从零认识进程 — 中下篇

news2025/1/16 0:59:49

在这里插入图片描述
送给大家一句话:
人一切的痛苦,本质上都是对自己无能的愤怒。而自律,恰恰是解决人生痛苦的根本途径。—— 王小波

从零认识进程

  • 1 进程优先级
    • 1.1 什么是优先级
    • 1.2 为什么要有优先级
    • 1.3 Linux优先级的特点 && 查看方式
    • 1.4 其他概念
  • 2 命令行参数
  • 3 环境变量
    • 3.1 什么是环境变量
    • 3.2 见见更多的环境变量
    • 3.3 整体理解
  • Thanks♪(・ω・)ノ谢谢阅读!!!
  • 下一篇文章见!!!

今天我们继续学习Linux的进程,上两篇文章我们认识了什么是进程,如何创建进程,进程状态。今天我们主要讲解 进程优先级和环境变量。

1 进程优先级

学习优先级需要了解:

  1. 什么是优先级
  2. 为什么要有优先级
  3. Linux优先级的特点 && 查看方式

1.1 什么是优先级

  • cpu资源分配的先后顺序,就是指进程的优先权(priority)。
  • 优先权高的进程有优先执行权利。配置进程优先权对多任务环境的linux很有用,可以改善系统性能。
  • 还可以把进程运行到指定的CPU上,这样一来,把不重要的进程安排到某个CPU,可以大大改善系统整体性能
    其实很简单,就是指定进程的获取资源的先后顺序(在运行队列中的相对顺序)。
    在task_struct模块中的内部字段 存在 int prio = ??,优先级的本质是一个数字,数字越小,优先级越高

1.2 为什么要有优先级

因为CPU的资源是有限的,不得不排队(就像食堂打饭一样),系统大部分情况进程是比较多的,但是硬件只有一套。

操作系统关于的调度好优先级的原则:分时操作系统,基本的公平。如果进程因为长时间不被调度,就造成的饥饿问题。
我们来简单查看一下进程的优先级:

  1 #include<stdio.h>
  2 #include<unistd.h>
  3 
  4 int main()
  5 {
  6   while(1){
  7     printf("I am a process , pid: %d\n",getpid());                                 
  8     sleep(1);
  9   }
 10 }
 11 

这里介绍一下一个新指令: ps -l
在这里插入图片描述
运行是运行了,但是没有我们现在的进程,这是因为ps -l默认是显示当前终端的进程,我们使用ps -al就可以查看全部的了。
在这里插入图片描述
其中:

  • UID : 代表执行者的身份
  • PID : 代表这个进程的代号
  • PPID :代表这个进程是由哪个进程发展衍生而来的,即父进程的PID
  • PRI : 进程优先级
  • NI : 进程优先级的修正数据,nice值,新的优先级 = 优先级 + nice值,达到对进程优先级的动态化处理(每次调用都可以更新nice值)

通过上图,可以看到我们进程(pid : 6000)的优先级是80。

1.3 Linux优先级的特点 && 查看方式

接下来我们来看如何查看修改优先级
用top命令更改已存在进程的nice:
进入top后按 r 输入进程PID –> 输入nice值,就可以修改nice值。来进行一下尝试:
在这里插入图片描述
可以看到,我们想要改到100,但是最终结果是99,这是为什么???
nice值不能随意调整,而是有范围的 [-20 , 19)共四十个数字(即四十 个 梯度)
我们通过ps-al查看的进程信息中,nice值就是最大值 19。
每次调整完优先级都是从80开始的,通过 新的优先级 = 优先级(80) + nice值进行计算。
注意:作为普通用户不能频繁调整优先级 想要多次调整需要root身份。

同时调整优先级也可以使用nicerenice命令,具体使用方法可以用那个男人 man 来进行查看(以renice 为例):

RENICE(1)                           User Commands                          RENICE(1)

NAME
       renice - alter priority of running processes

SYNOPSIS
       renice [-n] priority [-gpu] identifier...

DESCRIPTION
       renice  alters the scheduling priority of one or more running processes.  The
       first argument is the priority value to be used.   The  other  arguments  are
       interpreted as process IDs (by default), process group IDs, user IDs, or user
       names.  renice'ing a process group causes all processes in the process  group
       to have their scheduling priority altered.  renice'ing a user causes all pro‐
       cesses owned by the user to have their scheduling priority altered.
OPTIONS
       -n, --priority priority
              Specify the scheduling priority to be used for  the  process,  process
              group,  or  user.  Use of the option -n or --priority is optional, but
              when used it must be the first argument.

       -g, --pgrp pgid...
              Force the succeeding arguments to be interpreted as process group IDs.

       -u, --user name_or_uid...
              Force the succeeding arguments to be interpreted as usernames or UIDs.

       -p, --pid pid...
              Force the succeeding arguments to be interpreted as process  IDs  (the
              default).

       -h, --help
              Display a help text.

       -V, --version
              Display version information.


也就是使用 renice -n ? -p ?就可以完成修改优先级。

1.4 其他概念

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

2 命令行参数

命令行参数其实很简单,我们已经用过许多次了。类似 rm -f grep -v 等等都是命令行参数。那么这本质到底是什么呢???

我们先从main函数讲起,我们都知道main函数有参数int main( int argc, char* argv[ ]),(但实际我们经常不带,今天不管这个问题)
下面我们来看具体是什么作用:

    1 #include<stdio.h>
    2 #include<unistd.h>
    3 
W>  4 int main(int argc , char* argv[])//这里的报错先忽视
    5 {
    6   for(int i = 0; i < argc ;i++ )
    7   {
    8     printf("argv[%d]-> %s\n",i,argv[i]);                                         
    9   }
   10   return 0;
   11 }                                                            
   12  

来看运行效果:
在这里插入图片描述
实际上main函数的参数就是命令行的参数,也就是命令行字符串。前面的./myprocess是程序的路径和名称,后面的-a -b -c -d就是该进程匹配的选项。
那为什么要这么干呢???又是谁干的呢???
来看下面一段代码:

  1 #include<stdio.h>
  2 #include<unistd.h>
  3 #include<string.h>
  4 
  5 int main(int argc , char* argv[])
  6 {
  7   if(argc != 2)
  8   {
  9     printf("Usage:%s -[a,b,c,d]\n",argv[0]);
 10     return 1;
 11   }
 12 
 13   if(strcmp(argv[1],"-a") == 0)
 14   {
 15     printf("this is function1\n");
 16   }
 17 
 18   else if(strcmp(argv[1],"-b") == 0)
 19   {
 20     printf("this is function2\n");
 21   }
 22 
 23   else if(strcmp(argv[1],"-c") == 0)
 24   {
 25     printf("this is function3\n");
 26   }
 27 
 28   else if(strcmp(argv[1],"-d") == 0)
 29   {
 30     printf("this is function1\n");
 31   }                               
 32   else
 33   {
 34     printf("no this function!!!\n");
 35   }                                                                                
 36   return 0;                        
 37 }                                  
 38  

这样可以区分命令行输入了哪些信息,并执行相应功能。来看效果:

laik
为什么要有命令行参数???
本质:命令行参数的本质是交给我们不同程序的不同选项,用来定制不同功能,一般命令里会带许多选项

这样通过不同选项就执行程序的不同功能。这是不是有点像我们经常使用的哪些命令呢!?ll ls rm -f等等
那这个工作时是谁来干的呢???
我们进行一个小小的测试:

    1 #include<stdio.h>
    2 #include<unistd.h>
    3 #include<string.h>
    4 
    5 int g_val = 10000;
    6 
W>  7 int main(int argc , char* argv[])    
	8 {    
	9   printf("I am father process,pid : %d ,ppid: %d ,g_val: %d\n",getpid(),getppid(),g_val);
   10   sleep(5);
   11   
   12   pid_t id = fork();
   13   if(id == 0)
   14   {
   15     //child
   16     while(1)
   17     {
   18       printf("I am child process,pid: %d,ppid: %d,g_val: %d\n",getpid(),getppid(),g_val);
   19       sleep(1);   
   20     }
   21   }   
   22   else   
   23   {
   24     //father
   25        
   26       printf("I am father process,pid: %d,ppid: %d,g_val: %d\n",getpid(),getppid(),g_val);                                                                        
   27       sleep(1);                       
   28   }                                   
   29                                       
   30   return 0;                           
   31 }

来看效果:
在这里插入图片描述
可以看到子进程和父进程都成功使用了g_val,即父进程的数据默认可以被子进程看到并访问!!!
而其中的PID:10069是谁呢???
命令行中的程序都会变成进程,其实都是bash 的子进程!!!
所以那这个工作时是谁来干的呢???都是bash进行的,也就是main函数的参数是bash处理的

3 环境变量

3.1 什么是环境变量

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

系统中的很多配置,在我们登录Linux就已经被加载到bash进程中了。
来看一个(搜索路径的环境变量):
在这里插入图片描述
bash在执行命令的时候,需要先找到命令,因为未来要加载!!!
同样只要我们把我们写可执行程序拷贝到对应路径就可以不用再写./就可以执行。
当然这样太粗暴了,我们可以使用PATH = 路径这样就改变我们的环境变量了:
在这里插入图片描述

但是现在我们好多指令都不能正常使用了???啊???怎么办???原来的路径不一致,寻找不到了大部分指令。
这时候重启即可。
在这里插入图片描述
这样就恢复了,但是到底怎样才能把我们的程序路径加入进PATH呢???
应该使用PATH=$PATH : 路径即可。
在这里插入图片描述
这样就成功加入了!!!

注意 最开始的环境变量不是在内存中,而是在系统的对应的配置文件中
那配置文件在哪里呢???就在家目录的几个文件中
在这里插入图片描述

3.2 见见更多的环境变量

环境变量都是大写的英文字母,使用echo $名字即可
使用env就可以查看所有的环境变量:
在这里插入图片描述
这些都是环境变量!!!我们来认识其中几个:

  1. HOME:这个代表登录默认所处路径,即家目录
  2. PWD:这个会动态储存我们所在的目录
  3. SHELL:这个会默认启动命令行解释器,让我们可以输入命令
  4. HISTSIZE:这个是指历史命令的个数,意义是历史命令最多有多少条(一般是1000条)

等等都是环境变量
当然自己也可以定义环境变量。
使用export就可以进行:export THIS_IS_MY_ENV = hellobit这样就创建一个环境变量!!!
如果不加export 就会创建本地变量,与环境变量不同(需要对通信,多线程有一定了解才能理解)。

3.3 整体理解

    1 #include<stdio.h>
    2 #include<unistd.h>
    3 #include<string.h>
    4 
    5 int g_val = 10000;
    6 
    7 int main()
    8 {
    9   extern char** environ;
   10   for(int i = 0;environ[i];i++)
   11   {
   12     printf("env[%d]->%s\n",i,environ[i]);                                                                                                                                   
   13   }                                                                                                                                      
   14   return 0;                                                                                                                              
   15 }         

在这里插入图片描述
这样我们就通过C语言程序成功获取了环境变量。
然后因为我们的程序是一个进程,是bash的子进程,所以环境变量可以被子进程获取!!!
(环境变量在BASH中)

首先数据储存在磁盘中,运行时会加载到内存中,也就把环境变量存入内存中的bash/shell.
bash进程在启动的时候,默认会给子进程形成两张表:

  1. argv[ ] :命令行参数。
  2. env[ ] : 环境变量表,通过各种方式交给子进程。

环境变量具有系统级的全局属性,因为会被子进程继承下去!!!
esport , echo 等是内建命令,由bash执行,80%命令是bash创建的子进程实行

Thanks♪(・ω・)ノ谢谢阅读!!!

下一篇文章见!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1539523.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C语言】编译和链接----从源代码到可执行程序的转换【图文详解】

欢迎来CILMY23的博客喔&#xff0c;本篇为【C语言】文件操作揭秘&#xff1a;C语言中文件的顺序读写、随机读写、判断文件结束和文件缓冲区详细解析【图文详解】&#xff0c;感谢观看&#xff0c;支持的可以给个一键三连&#xff0c;点赞关注收藏。 前言 欢迎来到本篇博客&…

写作兔怎么用 #微信#其他#知识分享

写作兔是一款非常实用的论文写作工具&#xff0c;不仅具有查重和降重的功能&#xff0c;而且操作简单方便&#xff0c;使用起来非常便捷。那么&#xff0c;接下来就让我们一起来了解一下“写作兔怎么用”。 首先&#xff0c;要使用写作兔&#xff0c;你只需要在浏览器中输入写作…

背包dp模板

01背包 for (int i1; i<n; i) //物品 {for (int j1; j<V; j) //容积 { // 装得下 分为 1.装 2.不装if (j>v[i]) dp[j] max(tmp[j],tmp[j-v[i]]v[i]);else dp[j] tmp[j]; // 装不下第i个}for (int j1; j<V; j) tmp[j] dp[j]; //滚动数组 } 滚动数组 for (int …

网络安全知识核心之TCP与UDP区别

TCP 面向连接&#xff08;如打电话要先拨号建立连接&#xff09;提供可靠的服务;UDP 是无连接的&#xff0c;即发送数据之前不需要建立连接&#xff0c;;UDP 尽最大努力交付&#xff0c;即不保证可靠交付。&#xff08;由于 UDP 无需建立连接&#xff0c;因此 UDP 不会引入建立…

【漏洞复现】科立讯通信指挥调度平台editemedia.php sql注入漏洞

漏洞描述 在20240318之前的福建科立讯通信指挥调度平台中发现了一个漏洞。该漏洞被归类为关键级别,影响文件/api/client/editemedia.php的未知部分。通过操纵参数number/enterprise_uuid可导致SQL注入。攻击可能会远程发起。 免责声明 技术文章仅供参考,任何个人和组织使…

2024年全球生成人工智能全景图【中文】

2024年全球生成人工智能全景图【中文】 在过去的一年中&#xff0c;产生式人工智能&#xff08;GenAI&#xff09;无疑成为了全球各行各业的热门话题。特别是ChatGPT的发布&#xff0c;激发了公众对GenAI强烈的兴趣和激动&#xff0c;唤醒了我们对其变革潜力的认知。 虽然我们…

算法系列--动态规划--回文子串系列

&#x1f495;"我们好像在池塘的水底&#xff0c;从一个月亮走向另一个月亮。。"&#x1f495; 作者&#xff1a;Mylvzi 文章主要内容&#xff1a;算法系列–动态规划–回文子串系列 今天为大家带来的是算法系列--动态规划--回文子串系列(1),本文重点掌握如何快速判…

树,二叉树与堆

这里写目录标题 树树的概念树的相关概念树的表示 二叉树二叉树的概念满二叉树与完全二叉树二叉树的重要性质二叉树的存储结构 堆二叉树的顺序存储堆的概念堆的实现堆插入和删除数据 树 树的概念 树的概念&#xff1a; 树是一种非线性的数据结构&#xff0c;它是由n&#xff08…

链表oj测试题(上)

链表的申明&#xff1a; struct ListNode {int val;struct ListNode* next; }; 1.题1 删除指定元素 例如&#xff1a;链表1 2 6 3 4 5 6&#xff0c;然后选择删除元素6&#xff0c;返回的链表为1 2 3 4 5 。 代码演示&#xff1a; typedef struct ListNode ListNode;List…

01-java面试题八股文-----java基础——20题

文章目录 <font color"red">1、java语言有哪些特点&#xff1a;<font color"red">2、面向对象和面向过程的区别<font color"red">3、标识符的命名规则。<font color"red">4、八种基本数据类型的大小&#xff…

Elastic 线下 Meetup 将于 2024 年 3 月 30 号在武汉举办

2024 Elastic Meetup 武汉站活动&#xff0c;由 Elastic、腾讯、新智锦绣联合举办&#xff0c;现诚邀广大技术爱好者及开发者参加。 活动时间 2024年3月30日 13:30-18:00 活动地点 中国武汉 武汉市江夏区腾讯大道1号腾讯武汉研发中心一楼多功能厅 13:30-14:00 入场 活动流程…

关于调度算法,小林给出更好的例子(银行办理业务)

看的迷迷糊糊&#xff1f;那我拿去银行办业务的例子&#xff0c;把上面的调度算法串起来&#xff0c;你还不懂&#xff0c;你锤我&#xff01; 办理业务的客户相当于进程&#xff0c;银行窗口工作人员相当于 CPU。 现在&#xff0c;假设这个银行只有一个窗口&#xff08;单核 …

Prometheus Grafana 配置仪表板

#grafana# 其实grafana提供了丰富的Prometheus数据源的仪表板&#xff0c;基本上主流的都有&#xff0c;通过下面官方地址可查阅 Dashboards | Grafana Labs 这里举例说明&#xff0c;配置node_exporter仪表板 首先&#xff0c;在上面的网站搜索 node 可以查到蛮多的仪表板…

使用 Pytorch 和 Rasterio 的自定义地理空间数据加载器

地理空间数据在从遥感和城市规划到环境监测和灾害管理的各个领域发挥着至关重要的作用。在处理机器学习任务的地理空间数据时,准备自定义数据加载器对于有效加载、预处理和增强数据而不丢失其属性至关重要,特别是当输入图像具有超过 3 个波段时。 Rasterio确实是一个专门为有…

FPGA——DDR3的IP核

FPGA——DDR3的ip核 IP核配置基于MIG核代码基于AXI接口的DDR3 IP核配置 1 2 3 4 5 6 基于MIG核代码 控制MIG核的信号进行读写 module MIG_APP_Drive(input i_ui_clk ,input i_ui_rst ,input init_calib_…

SpringCloud Alibaba Nacos 服务注册和配置中心

一、前言 接下来是开展一系列的 SpringCloud 的学习之旅&#xff0c;从传统的模块之间调用&#xff0c;一步步的升级为 SpringCloud 模块之间的调用&#xff0c;此篇文章为第十二篇&#xff0c;即介绍 SpringCloud Alibaba Nacos 服务注册和配置中心。 二、Nacos 简介 2.1 为…

【DataWhale】灵境Agent开发——低代码创建AI智能体

灵境Agent开发——低代码创建AI智能体 3 灵境 Agent 低代码开发 ​ 低代码模式支持开发者通过编排工作流的方式快速构建智能体&#xff0c;您可以通过拖拽和组合模型、提示词、代码等模块&#xff0c;实现准确的、复杂的业务流程。 ​ 个人体验下来&#xff0c;目前这个低代…

华为OD机22道试题

华为OD机试题 2.查找小朋友的好朋友位置 在学校中&#xff0c;N 个小朋友站成一队&#xff0c;第 i 个小朋友的身高为 height[i]&#xff0c;第 i 个小朋友可以看到第一个比自己身高更高的小朋友j&#xff0c;那么 j 是 i 的好朋友 (要求&#xff1a;j>i) 。 请重新生成一个…

[运维] 可视化爬虫易采集-EasySpider(笔记)

一、下载 ​下载地址 下滑到Assets页面&#xff0c;选择下载 二、解压运 ​解压压缩包&#xff0c;打开文件夹 在此文件夹下打开Linux Terimal, 并输入以下命令运行软件&#xff1a; ./easy-spider.sh 注意软件运行过程中不要关闭terminal。 三、使用 1.开始 首先点击…

机器学习算法那些事 | 数据算法工程师必须掌握的5个Python库

本文来源公众号“OpenCV与AI深度学习”&#xff0c;仅用于学术分享&#xff0c;侵权删&#xff0c;干货满满。 原文链接&#xff1a;数据算法工程师必须掌握的5个Python库 如果你是一名初学者或中级机器学习工程师或数据科学家&#xff0c;这篇文章非常适合你。你已经选择了自…