<Linux>进程

进程

文章目录

  • 进程
    • PCB
    • pid与ppid
    • fork系统调用
    • 进程状态
    • 孤儿进程
    • 状态优先级
    • 环境变量
    • 进程地址空间
    • 虚拟地址

最直观的表示:启动一个软件,本质就是启动一个进程

PCB

PCB是Process Control Block的简称,是用来描述进程状态信息的数据结构。

进程运行时,是会在内存中存放该进程的代码+数据

因此:进程=代码和数据+PCB

这里的PCB结构是一种泛化的概念,在Linux下,PCB的实例化就是task_struct

pid与ppid

就像人有名字一样,进程也有自己的名字,进程的名字就是pid,ppid就是该进程的父进程的pid

fork系统调用

fork是创建进程的系统调用,给父进程返回子进程pid,给子进程返回0

#include<stdio.h>
#include<sys/types.h>

int main()
{
    pid_t id =fork();

    if(0!=id)
    {
        printf("我是父进程,pid:%d 我的子进程pid:%d\n",getpid(),id);
        //getpid()返回pid
    }
    else if(0==id)
    {

        printf("我是子进程,pid:%d 我的父进程pid:%d\n",getpid(),getppid());
        //getppid()返回父进程pid
    }
    return 0;
}

运行结果如下
在这里插入图片描述

注意

  • fork之后会有两个不同的执行流,至于父进程对于执行流先运行还是子进程对于执行流先运行不一定,这由操作系统的调度器决定
  • fork之后代码由父子进程共享

fork之后子进程会拷贝父进程的代码与数据给自己,操作系统会给子进程创建对应的PCB。

进程状态

  • 运行态,这里的运行态不单单指在cpu上运行才叫做运行态,在运行队列中排队等待调度器调度时也叫做运行态
  • 阻塞,阻塞态可以简单地理解为进程等待某种资源的释放或某种事件的发生,进程在这种状态下暂时停止执行,直到所需的资源可用或所等待的事件发生为止。
  • 挂起,内存不足时,操作系统会适当置换进程的代码与数据到磁盘,这种状态就叫做挂起状态。

Linux下的进程状态

  • R:就是上面运行态

验证R状态

#include <stdio.h>

int main()
{
    while (1)
    {
    }
    return 0;
}

运行起来之后写一段shell脚本来检测进程运行状态

while :;do  ps axj | head -1 && ps axj | grep test | grep -v grep ; sleep 1 ;echo "------------------"; done

在这里插入图片描述

R后面的‘+’表示这是一个前台进程

  • S:对于上面阻塞态,可被操作系统唤醒,可中断睡眠

验证S状态

#include <stdio.h>

int main()
{
    int a;
    scanf("%d", &a);
    return 0;
}

运行起来之后,系统会等着你输入,这时候不要输入,去查看进程状态

在这里插入图片描述

  • D:深度睡眠,不可被中断,不可被被动唤醒

不做验证

  • T:暂停状态

在用一些调试工具调试时可以验证,这里不做验证

  • Z:僵尸状态,进程已经退出,但是还不允许操作系统释放,处于被检测的状态。维持该状态是为了父进程/操作系统来回收

验证Z状态

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>
int main()
{
    pid_t id = fork();
    if (0 == id)
    {
        int cnt = 3;
        while (cnt--)
        {
            printf("我是子进程,pid:%d,ppid:%d\n", getpid(), getppid());
            sleep(1);
        }
        exit(0);
    }
    else
    {
        while (1)
        {
            printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());
            sleep(1);
        }
    }
    return 0;
}

可以看到,前面三秒都是S装态,这里S状态是因为sleep了一秒钟,第四秒开始子进程变成僵尸状态,等待着父进程或者操作系统来回收,也就是对应的Z状态

在这里插入图片描述

子进程的进程名后面出现了‘defunct’,即为失效的,即僵尸状态

在这里插入图片描述

打印输出也只剩父进程

如何解决僵尸进程?在我的另一篇博客:中有详细解读,感兴趣的朋友可以自行阅读。

  • X:终止状态

孤儿进程

父进程先退出,子进程就被称为孤儿进程,此时该进程就会被init进程(pid为1)领养,由init回收

验证孤儿进程

// 验证孤儿进程
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>

int main()
{
    pid_t id = fork();
    if (0 == id)
    {
        while (1)
        {
            printf("我是子进程,pid:%d,ppid:%d\n", getpid(), getppid());
            sleep(1);
        }
    }
    else
    {
        int cnt = 3;
        while (cnt--)
        {
            printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());
            sleep(1);
        }
        exit(0);
    }
    return 0;
}

在这里插入图片描述

可以看到,前三秒父进程和子进程同时在跑,但是3秒后父进程退出,子进程被1号进程领养

状态优先级

进程优先级即cpu资源分配的先后顺序

查看进程优先级命令

ps -l

在这里插入图片描述

  • PRI

代表进程优先级,数值越小,优先级越高

  • NI

nice值,用来更改进程优先级,取值范围[-20,19]

PRI=PRI+NI

所以想要修改进程优先级,就要通过修改nice值来进而修改进程优先级

修改进程优先级命令

top

输入‘r’后输入进程pid,随后修改nice值即可

环境变量

先来看现象

我们写一个打印的程序

#include <stdio.h>
int main()
{
    printf("hello linux\n");
    return 0;
}

在这里插入图片描述

使用系统自带的命令时,不用带路径,直接可以运行,但是我们自己写的程序如果不带路径就会报错。

这就和环境变量有关

查看环境变量

echo $PATH

在这里插入图片描述

这里维护了大量的路径,路径之间用冒号’:‘隔开,我们使用的诸如’ls’,'pwd’的系统命令就在这些路径当中,我们可以查看一下ls所处的路径

which ls

在这里插入图片描述

所以我们运行自己的程序时,shell会在这些路径中去查找’myproc’,没有找到就会显示’command not found’

如果不想带路径运行自己的程序,可以将可执行程序所在的路径导入到系统环境变量中,即可像系统指令一样执行自己写的程序。

export PATH=$PATH:程序所在路径

在这里插入图片描述

导入之后再查看系统环境变量,这时程序即可直接运行,不用带路径

但是,这种方式只在本次对话当中有用,下次连接系统时不会存在这个路径,如果需要永久保存,需要修改环境变量配置文件。

但是呢,环境变量不止PATH,还有HOME

echo $HOME

在这里插入图片描述

SHELL环境变量

echo $SHELL

在这里插入图片描述

显示所有环境变量

env
  • 子进程的环境变量继承自父进程,往上层去找最终就是bash的环境变量
  • 环境变量具有全局属性,体现在可以被子进程继承

进程地址空间

在c/c++程序员眼中,程序地址空间如下

在这里插入图片描述

#include <stdio.h>
#include <stdlib.h>
int uninit_global;
int init_global = 10;

int main(int argc, char *argv[], char *env[])
{
    const char *str = "read only";

    int *heap1 = (int *)malloc(sizeof(int));
    int *heap2 = (int *)malloc(sizeof(int));

    printf("code address:%p\n", main);
    printf("read only address:%p\n", str);
    printf("init_global address:%p\n", &init_global);
    printf("uninit_global address:%p\n", &uninit_global);
    printf("head address:%p\n", heap1);
    printf("head address:%p\n", heap2);

    printf("stack address:%p\n", &heap1);
    printf("stack address:%p\n", &heap2);

    int i = 0;
    for (i = 0; i < argc; i++)
    {
        printf("argv[%d] address:%p\n", i, &argv[i]);
    }
    for (i = 0; env[i]; i++)
    {
        printf("env[%d]:%p\n", i, env[i]);
    }

    return 0;
}

输入命令运行结果如下

在这里插入图片描述

虚拟地址

为什么叫虚拟地址空间?难道上面我们讲的进程地址空间都是虚拟的?假的? 没错!!就是假的!!

如果直接让用户对真实的 物理空间进行读写访问,其实是非常危险的动作,因此,操作系统不会让你直接访问真实的物理地址。

操作系统表面上告诉进程,计算机的地址空间全是你的,但是其实不然。

先来一段代码

#include <stdio.h>
#include <sys/types.h>

int global = 10;

int main()
{
    pid_t id = fork();
    if (0 == id)
    {
        printf("global_val:%d---%p\n", global, &global);
    }
    else
    {
        global = 5;
        printf("global_val:%d---%p\n", global, &global);
    }
    return 0;
}

运行结果如下

在这里插入图片描述

出现了非常神奇的现象?同一个地址为什么对应的值不同???

我们知道,父进程在创建子进程之后会给子进程拷贝一份父进程的代码与数据,也会给子进程创建对应的程序地址空间,而这个程序地址空间就属于虚拟地址,需要页表哈希索引来找到对应的真实的物理地址,子进程拷贝的虚拟地址与父进程对应的虚拟地址一样,所以出现了地址一样,值却不一样的现象。解析图如下

在这里插入图片描述

上述程序因为拷贝了父进程的代码与数据,拷贝的地址属于虚拟地址,经过页表置换后才能找到真正的物理地址。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

STM32学习记录(五)————外部中断EXTI

文章目录 前言一、外部中断EXTI基础知识1.外部中断介绍2.外部中断框架2.1AFIO2. 2.STM32外部中断机制框架 总结 前言 一个学习STM32的小白~ 有问题私信或评论区指出~ 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、外部中断EXTI基础知识 1.外部中…

CSS加载动画1

3个圈圈加载的动画 CSS结构 #app-loading {position: relative;top: 45vh;margin: 0 auto;color: #409eff;font-size: 12px;}#app-loading,#app-loading::before,#app-loading::after {width: 2em;height: 2em;border-radius: 50%;animation: 2s ease-in-out infinite app-loa…

GraphQL(7):ConstructingTypes

1 使用GraphQLObjectType 定义type&#xff08;类型&#xff09; 不使用ConstructingTypes定义方式如下&#xff1a; 使用ConstructingTypes定义方式如下&#xff1a; 更接近于构造函数方式 var AccountType new graphql.GraphQLObjectType({name: Account,fields: {name: …

使用Redis将单机登录改为分布式登录

使用Redis将单机登录改为分布式登录 1. 背景 ​ 现在大多数的应用程序登录的方式都是必须满足分布式登录的效果&#xff0c;比如我们在一个客户端登录之后可以在另一个客户端上面共享当前用户的信息&#xff0c;这样在另一个客户端登录的时候就不用用户再次输入自己的账号密码…

Python 全栈系列253 再梳理flask-celery的搭建

说明 最近做了几个实验&#xff0c;将结论梳理一下&#xff0c;方便以后翻看。 1 flask-celery 主要用于数据流的同步任务&#xff0c;其执行由flask-aps发起&#xff0c;基于IO并发的方法&#xff0c;达到资源的高效利用&#xff0c;满足业务上的需求。2 目前部署环境有算网…

string类的使用手册

1.构造函数 补充&#xff1a;npos&#xff1a;size_t类型数据的最大值 default (1) string(); 构造空的string类对象 copy (2) string (const string& str); 拷贝构造函数&#xff08;深拷贝&#xff09; substring (3) string (const string& str, size_t pos, size_…

pytest + yaml 框架 - 65.Pycharm 设置 yaml 格式用例模板,高效写用例

前言 初学者对yaml 格式不太熟悉,自己写yaml用例的时候,总是格式对不齐,或者有些关键字会忘记。 于是我们可以在pycharm上设置用例模块,通过快捷方式调用出对应的模块,达到高效写用例的目的。 pycharm设置用例模板 File - Settings Live Templates - python 点 + 号…

【react小项目】bmi-calculator

bmi-calculator 目录 bmi-calculator初始化项目01大致布局01代码 02完善样式02代码 03输入信息模块03代码 04 使用图表04代码 05详细记录信息渲染05代码 06 让数据变成响应式的06-1输入框的数据处理06-2图表&#xff0c;和记录信息的区域数据处理 07 删除功能&#xff0c;撤销功…

521. 最长特殊序列 Ⅰ

题目 给你两个字符串 a 和 b&#xff0c;请返回这两个字符串中最长的特殊序列的长度。如果不存在&#xff0c;则返回 -1。 「最长特殊序列」定义如下&#xff1a;该序列为某字符串独有的最长子序列&#xff08;即不能是其他字符串的子序列&#xff09;。 字符串 s 的子序列是…

介绍 Whisper 模型

介绍 Whisper 模型 Whisper 是一个通用的语音识别模型。它在大规模多样化的音频数据集上进行训练&#xff0c;并且能够执行多任务处理&#xff0c;包括多语言语音识别、语音翻译和语言识别。 核心方法 Whisper 使用的是 Transformer 序列到序列模型&#xff0c;训练于多种语…

vue项目问题汇总

1.el-select&#xff1a; 下拉框显示到了top:-2183px , 添加属性 :popper-append-to-body"false" 2. el-upload: 选过的文件在使用过后记得清空&#xff0c;因为如果有limit1的时候&#xff0c;没有清空会导致不触发onchange 使用自定义上传方法http-request的时…

基于android开发平台的聊天软件实现(论文+源码)_kaic

摘要&#xff1a;互联网时代的到来使得手机通讯变得更为普及和强大&#xff0c;人们可以随时随地地进行交流。由于工作的繁忙以及生活节奏的加快&#xff0c;人们无法有更多时间展开面对面的交谈&#xff0c;导致在线聊天软件的使用更加频繁&#xff0c;所以本文尝试设计了一款…

Excel报表

(Apache POI) 入门案例 P164 使用POI需要导入下面2个坐标&#xff1a; <dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId> </dependency> <dependency><groupId>org.apache.poi</groupId>&…

docker-compose部署FastDFS分布式文件系统

文章目录 一、技术选型二、fastDFS组成部分三、docker-compose文件四、客户端nginx配置五、存储器spring Boot集成参考文献 一、技术选型 还有一个更好的google FS&#xff08;但是他不开源&#xff0c;我也没找到社区版一类的可以不要钱使用的&#xff09;。 最后考虑到我们存…

AD学习记录

1. 负信号&#xff1a; \WR或者W\R\ 2.快捷键&#xff1a; MMS VGS X/W CTLRW原理图画总连接线&#xff0c;shift快速复制 TAA管理器&#xff0c;TG封装管理器 3. 选中后按住ctlr进行位移 4.原理图里切换原理图库&#xff1a; 5.重要警报&#xff1a;&#xff0…

CleanMyMacX4.15.4如何优化苹果电脑系统缓存,告别MacBook卡顿,提升mac电脑性能

你是否曾为苹果电脑存储空间不够而烦恼&#xff1f;是否曾因系统运行缓慢而苦恼&#xff1f;别担心&#xff0c;今天我要给大家种草一个神器——CleanMyMac&#xff01;这款软件可以帮助你轻松解决苹果电脑的种种问题&#xff0c;让你的电脑焕然一新&#xff01; 让我来给大家介…

论文学习day01

1.自我反思的检索增强生成&#xff08;SELF-RAG&#xff09; 1.文章出处&#xff1a; Chan, C., Xu, C., Yuan, R., Luo, H., Xue, W., Guo, Y., & Fu, J. (2024). RQ-RAG: Learning to Refine Queries for Retrieval Augmented Generation. ArXiv, abs/2404.00610. 2.摘…

使用消息队列(MQ)实现MySQL持久化存储与MySQL server has gone away问题解决

在现代应用程序开发中&#xff0c;消息队列&#xff08;MQ&#xff09;扮演着重要的角色。它们可以帮助我们解决异步通信和解耦系统组件之间的依赖关系。而其中一个常见的需求是将消息队列中的数据持久化到数据库中&#xff0c;以确保数据的安全性和可靠性。在本文中&#xff0…

java第二十四课 —— super 关键字 | 方法重写

super 关键字 基本介绍 super 代表父类的引用&#xff0c;用于访问父类的属性、方法、构造器。 基本语法 访问父类的属性&#xff0c;但不能访问父类的 private 属性。 super.属性名; 访问父类的方法&#xff0c;不能访问父类的 private 方法。 super.方法名(参数列表); 访…

Java的一些内容

transient的作用 transient是Java语言的关键字&#xff0c;用来表示一个成员变量不是该对象序列化的一部分。当一个对象被序列化的时候&#xff0c;transient型变量的值不包括在序列化的结果中。而非transient型的变量是被包括进去的。 注意static修饰的静态变量天然就是不可序…