[linux]--关于进程概念(下)

目录

孤儿进程

环境变量

将程序放到PATH路径下

设置PATH环境变量

设置别名

环境变量相关的命令

 环境变量的组织方式​编辑

通过系统调用获取环境变量

环境变量通常是具有全局属性的

进程优先级

查看系统进程

用top命令更改已存在进程的nice:

程序地址空间


我们上一篇讲了僵尸进程,知道如果子进程退出没有给到父进程返回码,子进程就会被系统维护,一直处于僵尸状态Z,那么,如果是我们的父进程提前退出了而不是休眠,子进程再退出,陷入Z状态,有没有处理方案呢?

孤儿进程

父进程如果提前退出,那么子进程后退出,进入Z之后,那该如何处理呢?

父进程先退出,子进程就称之为“孤儿进程”

孤儿进程被1号init进程领养,当然要有init进程回收喽。

#include <stdio.h>
 #include <unistd.h>
 #include <stdlib.h>
 
int main()
 {
    pid_t id = fork();
    if(id < 0){
        perror("fork");
        return 1;
    }
    else if(id == 0){//child
        printf("I am child, pid : %d\n", getpid());
        sleep(10);
    }else{//parent
        printf("I am parent, pid: %d\n", getpid());
        sleep(3);
        exit(0);
    }
    return 0;
 }

 

这段命令是一个简单的 Bash 脚本,让我来逐步解释一下:

  1. while :; do ... done:这是一个无限循环结构,其中 : 表示一个永远返回 true 的条件,因此这个循环会无限执行。

  2. ps axj | grep mytest | grep -v grep:这部分命令的作用是通过 ps 命令查看当前系统中所有进程的状态,并使用 grep 命令过滤出包含 "mytest" 字符串的进程。第二个 grep -v grep 的作用是去除含有 "grep" 字符串的行,避免误判。

  3. sleep 1:这是让脚本暂停 1 秒,然后再继续执行下一次循环。

  4. echo "##########":输出一行 ##########,用来做分隔符或者标记输出结果。

作用是不断地检查系统中是否存在名为 "mytest" 的进程,每隔 1 秒输出一次检查结果,并用分隔符标记不同的输出

 也就是说,当我们的父进程先退出了,操作系统会把子进程接管过来,避免产生内存泄漏。

跟上一篇讲的僵尸进程的原理是差不多的,只是父进程的状态差异。

环境变量

概念

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

 怎么理解呢,我们linux下通常来讲就是一堆堆的文件,指令还有函数这些,都是一个个的文件,我们yum install从远程服务端口下载的指令,比如tree,也是被放在了系统下的一个路径里面

我给大家看一下,比如ls

一般来说系统级的指令都在/usr/bin路径下,大家可以自己看一下

所以说,我们的指令其实是一个个的文件,包括我们的下载的指令,那么,这跟环境变量有什么关系呢,环境变量,就是在我们的linux下,查找我们的指令以及文件所在的路径,然后给到操作系统去运行

常见环境变量

PATH : 指定命令的搜索路径

HOME : 指定用户的主工作目录(即用户登陆到Linux系统中时,默认的目录)

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

查看环境变量方法     echo $NAME    //NAME:你的环境变量名称

 我们先写个简单的代码看看path的作用

#include <stdio.h>

int main()

{

printf("hello world!\n");

return 0;

}

我们看看./hello 执行和直接  hello执行的区别

 

这里的报错是因为path没有找到我们文件的路径,在我们平时运行的时候,我们都是./加文件名运行的,而ls,touch这些指令则是可以直接运行

这是因为我们的path是系统的环境变量,而这些指令保存在我们的path环境变量下,path通过路径寻找到我们对应的指令,然后执行这个指令文件的功能,按照这个说法,我们的文件也是可以直接运行的,只要我们能够加载到path指定的路径下。

 大家有兴趣自己试一下,我装了半天,电脑有点小问题

将我们的程序所在路径加入环境变量PATH当中, export PATH=$PATH:hello程序所在路径

 echo $PATH查看我们的路径是否包含在下面

 对比./hello执行和之间hello执行

为什么有些指令可以直接执行,不需要带路径,而我们的二进制程序需要带路径才能执行? 将我们的程序所在路径加入环境变量PATH当中, export PATH=$PATH:hello程序所在路径

还有什么方法可以不用带路径,直接就可以运行呢?

将程序放到PATH路径下

第一种方法就是将我们自己的程序放到PATH中的路径中去,这样在shell输入hello时,也能找到,例如我们将其放在/bin目录下:

$ hello
hello world
$ whereis hello
hello: /bin/hello

也就是说,如果你的程序安装在了PATH指定的路径,就需要配置PATH环境变量,在命令行输入就可以直接找到了。

设置PATH环境变量

那么如果想在指定的目录能够直接运行呢?很简单,那就是添加环境变量,例如将当前路径加入到PATH中:

$ PATH=$PATH:./   #这种方式只在当前shell有效,所有shell生效可修改/etc/profile文件
$ hello
hello world
设置别名

例如:

$ alias hello="/temp/hello"
$ hello
hello world

以上三种方法都可以达到目的。

测试HOME

1. 用root和普通用户,分别执行 echo $HOME ,对比差异

2.执行 cd ~; pwd ,对应 ~ 和 HOME的关系 

 我们可以看到,两个其实是差不多的,取决于你在哪个用户环境下使用而已。

环境变量相关的命令

1.echo: 显示某个环境变量值
2. export: 设置一个新的环境变量
3. env: 显示所有环境变量
4. unset: 清除环境变量
5. set: 显示本地定义的shell变量和环境变量

 环境变量的组织方式

每个程序都会收到一张环境表,环境表是一个字符指针数组,每个指针指向一个以’\0’结尾的环境字符串 

通过代码如何获取环境变量

命令行第三个参数(就是环境参数)

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

通过第三方变量environ获取,指向我们的环境表,相当于二级指针

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

 libc中定义的全局变量environ指向环境变量表,environ没有包含在任何头文件中,所以在使用时 要用extern声明。

通过系统调用获取环境变量

#include <stdio.h>
 #include <stdlib.h>
 
int main()
 {
    printf("%s\n", getenv("PATH"));
    return 0;
 }

常用getenv和putenv函数来访问特定的环境变量。

环境变量通常是具有全局属性的

环境变量通常具有全局属性,可以被子进程继承下去

#include <stdio.h>
 #include <stdlib.h>
 
int main()
 {
    char * env = getenv("MYENV");
    if(env){
        printf("%s\n", env);
    }
    return 0;
 }

这个时候我们直接查看,发现没有结果,说明该环境变量根本不存在

导出环境变量
export MYENV="hello world"
再次运行程序,发现结果有了!说明:环境变量是可以被子进程继承下去的!

如果只进行MYENV=“helloworld” ,不调用export导出,在用我们的程序查看,会有什么结果?为什 么? 

如果只是直接设置环境变量 MYENV="helloworld" 而没有使用 export 命令导出该环境变量,然后再运行程序查看环境变量的取值,结果可能会是程序无法获取到正确的环境变量值。这是因为在不使用 export 命令导出环境变量时,该环境变量只在当前 shell 进程中有效,而不会被传递给程序的执行环境。

具体来说,当你直接设置环境变量 MYENV="helloworld" 时,该环境变量只会存在于当前 shell 进程中,而不会被传递给程序的执行环境。因此,当你运行程序时,程序尝试通过 getenv("MYENV") 函数获取环境变量的值时,由于该环境变量并未在程序的执行环境中被正确设置,程序将无法获得正确的环境变量值,结果可能是空值或者 null。

为了让程序能够正确获取环境变量的值,需要使用 export 命令将环境变量导出,使其成为子进程可见的环境变量。只有通过 export 命令导出的环境变量才能被程序正确识别和获取。

进程优先级

概念

cpu资源分配的先后顺序,就是指进程的优先权(priority)。

优先权高的进程有优先执行权利。

配置进程优先权对多任务环境的linux很有用,可以改善系统性能。

还可以把进程运行到指定的CPU上,这样一来,把不重要的进程安排到某个CPU,可以大大改善系统整 体性能。

一般来说,我们现在的电脑环境下,所启动进行的进程优先级都是一样的,所以这个部分主要是理解为主。

查看系统进程

在linux或者unix系统中,用ps –l命令则会类似输出以下几个内容:

这几个信息跟我们的进程优先级可以说有直接的关系

UID : 代表执行者的身份

PID : 代表这个进程的代号

PPID :代表这个进程是由哪个进程发展衍生而来的,亦即父进程的代号

PRI :代表这个进程可被执行的优先级,其值越小越早被执行

NI :代表这个进程的nice值

PRI and  NI

PRI也还是比较好理解的,即进程的优先级,或者通俗点说就是程序被CPU执行的先后顺序,此值越小 进程的优先级别越高 那NI呢?就是我们所要说的nice值了,其表示进程可被执行的优先级的修正数值 PRI值越小越快被执行,那么加入nice值后,将会使得PRI变为:PRI(new)=PRI(old)+nice 这样,当nice值为负值的时候,那么该程序将会优先级值将变小,即其优先级会变高,则其越快被执行 所以,调整进程优先级,在Linux下,就是调整进程nice值 nice其取值范围是-20至19,一共40个级别。 

需要强调一点的是,进程的nice值不是进程的优先级,他们不是一个概念,但是进程nice值会影响到进 程的优先级变化。 可以理解nice值是进程优先级的修正修正数据

用top命令更改已存在进程的nice:

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

比如说我要修改22进程的nice值,就是top进入我们的这歌页面,然后r一下,就会有命令窗口,输入22,然后回车,这个时候呢我写入一个60吧,方便我们学习

 我们的nice最大是19,我们原来是-20,所以60-40=20,那么pri是不是20+19=39呢?

并不是,pri每个人的峰值可能不太一样,我的是39最大,pri=之前的pri加nice值,之前的pri是0,加nice值应该是19啊!为什么呢,因为这里有一个bug,pri更改默认是20,所以我们才会看到39这个数值

我现在把60减掉再用40给大家演示一下,一样是39的pri

 

我们使用的时候注意一下就好了,一般也不会更改优先级

相关知识点

竞争性: 系统进程数目众多,而CPU资源只有少量,甚至1个,所以进程之间是具有竞争属性的。为了高 效完成任务,更合理竞争相关资源,便具有了优先级

独立性: 多进程运行,需要独享各种资源,多进程运行期间互不干扰

并行: 多个进程在多个CPU下分别,同时进行运行,这称之为并行

并发: 多个进程在一个CPU下采用进程切换的方式,在一段时间之内,让多个进程都得以推进,称之为 并发

程序地址空间

我们看一下c语言下的地址空间布局

可能我们对他并不理解!

#include <stdio.h>
 #include <unistd.h>
 #include <stdlib.h>
 
int g_val = 0;
 
int main()
 {
    pid_t id = fork();
    if(id < 0){
        perror("fork");
        return 0;
    }
    else if(id == 0){ //child
  printf("child[%d]: %d : %p\n", getpid(), g_val, &g_val);
    }else{ //parent
        printf("parent[%d]: %d : %p\n", getpid(), g_val, &g_val);
    }
    sleep(1);
    return 0;
 }

我们试一下从代码的角度去理解

有没有发现,地址是一样的啊

输出出来的变量值和地址是一模一样的,很好理解呀,因为子进程按照父进程为模版,父子并没有对变 量进行进行任何修改。

我们继续看,小改一下

我们发现,父子进程,输出地址是一致的,但是变量内容不一样!能得出如下结论:

变量内容不一样,所以父子进程输出的变量绝对不是同一个变量

但地址值是一样的,说明,该地址绝对不是物理地址!

在Linux地址下,这种地址叫做 虚拟地址

我们在用C/C++语言所看到的地址,全部都是虚拟地址!

物理地址,用户一概看不到,由OS统一管理

OS必须负责将 虚拟地址 转化成 物理地址 。

所以之前说‘程序的地址空间’是不准确的,准确的应该说成 进程地址空间

同一个变量,地址相同,其实是虚拟地址相同,内容不同其实是被映射到了 不同的物理地址!

 


写的不太好,表达能力确实不行,我会尽量除了我能看明白再写清楚的,阿里嘎都~

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

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

相关文章

(附源码)基于SSM的校园兼职系统设计与实现

前言 &#x1f497;博主介绍&#xff1a;✌专注于Java、小程序技术领域和毕业项目实战✌&#x1f497; &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅&#x1f447;&#x1f3fb; 2024年Java精品实战案例《100套》 &#x1f345;文末获取源码联系&#x1f345; &#x1f31…

uniapp ios证书失效

前面是按照网上查找的方法 作者大大的地址 1、一个ios账户&#xff08;688付费版&#xff09; 2、登录 Apple Developer 3、创建Identifiers ps&#xff1a;创建时需继承苹果的sdk&#xff0c;只需要一个就行 点击continue再点击Register即可 4、创建.cer证书 &…

tcp seq ack

seq&#xff08;Sequence Number&#xff09;&#xff1a;32bits&#xff0c;表示这个tcp包的序列号。tcp协议拼凑接收到的数据包时&#xff0c;根据seq来确定顺序&#xff0c;并且能够确定是否有数据包丢失。 ack&#xff08;Acknowledgment Number&#xff09;&#xff1a;3…

安卓开发日记:实现APP重启逻辑,适用于热更后重启游戏进行加载

可根据合适的弹窗搭配使用重启逻辑&#xff0c;建议使用在热更包加载后使用&#xff0c;帮助部分热更后未及时生效的逻辑范围首先&#xff0c;在逻辑调用Activity类中创建一个成员变量&#xff0c;给后续逻辑接口直接使用 如下 public class MainActivity extends Activity {…

想当初级爬虫工程师,需要把爬虫学到什么程度?

这篇文章会说说我自己的心得体验&#xff0c;关于爬虫、关于工作&#xff0c;仅供参考。 学到哪种程度 暂且把目标定位初级爬虫工程师&#xff0c;简单列一下吧&#xff1a; &#xff08;必要部分&#xff09; 语言选择&#xff1a;一般是了解Python、Java、Golang之一 熟悉…

SCI一区 | Matlab实现SSA-TCN-BiGRU-Attention麻雀算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测

SCI一区 | Matlab实现SSA-TCN-BiGRU-Attention麻雀算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测 目录 SCI一区 | Matlab实现SSA-TCN-BiGRU-Attention麻雀算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测预测效果基本介绍模型描述程序…

Vue2在一个页面内动态切换菜单显示对应的路由组件

项目的需求是在一个页面内动态获取导航菜单&#xff0c;导航菜单切换的时候显示对应的路由页面&#xff0c;类似于tab切换的形式&#xff0c;切换的导航菜单和页面左侧导航菜单是同一个路由组件&#xff0c;只是放到了一个页面上&#xff0c;显示的个数不同&#xff0c;所有是动…

基于Python的Django开发接口框架搭建

1、安装Django&#xff0c;如图1 命令行执行&#xff1a;pip install django 2、创建Django项目&#xff0c;如图2 django-admin startproject imooc 3、创建Django的应用项目&#xff0c;如图3 必须进入项目文件下执行命令&#xff1a; python3 manage.py startapp app 4、…

Python基础学习笔记(一)

Python简介 Python 语言是一种跨平台、开源、免费、解释型、面向对象、动态数据类型的高级程序设计语言。早期版本的 Python 被称作是 Python1&#xff1b;Python2 最后一个版本是 2.7&#xff1b;Python3 是目前最活跃的版 本&#xff0c;基本上新开发的 Python 代码都会支持…

线程池实现“线程复用”的原理

线程池实现“线程复用”的原理 学习线程复用的原理&#xff0c;以及对线程池的 execute 这个非常重要的方法进行源码解析。 线程复用原理 我们知道线程池会使用固定数量或可变数量的线程来执行任务&#xff0c;但无论是固定数量或可变数量的线程&#xff0c;其线程数量都远远…

27-4 文件上传漏洞 - 黑名单绕过

环境准备:构建完善的安全渗透测试环境:推荐工具、资源和下载链接_渗透测试靶机下载-CSDN博客 一、黑名单绕过和黑白名单机制: 黑名单:黑名单中的文件不允许通过。白名单:白名单中的文件允许通过。二、黑白名单判断: 当输入一串后缀如"sfahkfhakj"时,黑名单不…

2024/03/21(网络编程·day7)

一、思维导图 二、 //定义删除函数 int do_delete(sqlite3 *ppDb) {int del_numb0;printf("请输入要删除的学生的学号:");scanf("%d",&del_numb);getchar();//准备sql语句char sql[128]"select *from Stu";sprintf(sql,"delete from …

贾志杰“大前端”系列著作出版发行

杰哥著作《VueSpringBoot前后端分离开发实战》2021年出版以来&#xff0c;累计发行2.6万册&#xff0c;受到广大读者热捧。后应读者要求&#xff0c;受出版社再次邀请&#xff0c;“大前端”系列之《剑指大前端全栈工程师》、《前端三剑客》由清华大学出版社陆续出版发行。系列…

敢为天下先!深圳市全力推动鸿蒙生态发展……程序员

3月19日&#xff0c;鸿蒙生态创新中心揭幕仪式在深圳正式举行。鸿蒙生态创新中心的建立是为构建先进完整、自主研发的鸿蒙生态体系&#xff0c;将深圳打造为鸿蒙生态策源地、集聚区的具体举措&#xff0c;也是推动我国关键核心技术高水平自立自强、数字经济高质量发展、保障国家…

有哪些工具可以替代Gitbook?这篇文章告诉你

你是否曾经在搜索在线文档创建和共享工具时&#xff0c;遇到了Gitbook? Gitbook 是一个相当出色的工具&#xff0c;具有强大的编辑和发布功能&#xff0c;但也有其不足之处&#xff0c;如使用起来有一定的技术要求&#xff0c;入门门槛较高等。如果你正在寻找Gitbook的替代品&…

CSS问题精粹1

1.关于消除<li>列表前的符号 我相信很多人在初学CSS时会遇到该问题&#xff0c;无论是创作导航&#xff0c;还是列表&#xff0c;前面都会有个黑点点或其它符号。 解决该问题其实很简单 采用list-style-type:none或list-style:none直接解决 如果你想更换前面的黑点点&a…

java怎么做带进度条的上传

在Java中实现带进度条的文件上传功能通常涉及到前后端的配合工作。前端负责收集文件并展示上传进度&#xff0c;后端负责接收和处理文件&#xff0c;并提供进度信息给前端。 前端部分&#xff1a; HTML&#xff1a;创建文件输入控件和进度条元素。 <input type"file…

QT界面制作

#include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);this->setWindowFlag(Qt::FramelessWindowHint);//接收动图QMovie *mv new QMovie(":/pictrue/th.gif…

springboot企业级抽奖项目业务一(登录模块)

开发流程 该业务基于rouyi生成好了mapper和service的代码&#xff0c;现在需要在controller层写接口 实际操作流程&#xff1a; 看接口文档一>controller里定义函数一>看给出的工具类一>补全controller里的函数一>运行测试 接口文档 在登录模块有登录和登出方…

(附源码)基于Spring Boot + Vue的校园综合信息服务平台设计与实现

前言 &#x1f497;博主介绍&#xff1a;✌专注于Java、小程序技术领域和毕业项目实战✌&#x1f497; &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅&#x1f447;&#x1f3fb; 2024年Java精品实战案例《100套》 &#x1f345;文末获取源码联系&#x1f345; &#x1f31…