Linux--进程概念

1.基本概念与基本操作

• 课本概念:程序的⼀个执⾏实例,正在执⾏的程序等
• 内核观点:担当分配系统资源(CPU时间,内存)的实体。

2 描述进程-PCB

基本概念
• 进程信息被放在⼀个叫做进程控制块的数据结构中,可以理解为进程属性的集合。
• 课本上称之为PCB(process control block),Linux操作系统下的PCB是: task_struct

task_struct-PCB的⼀种

• 在Linux中描述进程的结构体叫做task_struct。
• task_struct是Linux内核的⼀种数据结构,它会被装载到RAM(内存)⾥并且包含着进程的信息。

3.task_ struct

内容分类

• 标⽰符: 描述本进程的唯⼀标⽰符,⽤来区别其他进程。
• 状态: 任务状态,退出代码,退出信号等。
• 优先级: 相对于其他进程的优先级。
• 程序计数器: 程序中即将被执⾏的下⼀条指令的地址。
• 内存指针: 包括程序代码和进程相关数据的指针,还有和其他进程共享的内存块的指针
• 上下⽂数据: 进程执⾏时处理器的寄存器中的数据[休学例⼦,要加图CPU,寄存器]。
• I∕O状态信息: 包括显⽰的I/O请求,分配给进程的I∕O设备和被进程使⽤的⽂件列表。
• 记账信息: 可能包括处理器时间总和,使⽤的时钟数总和,时间限制,记账号等。
• 其他信息

这里重点关注我们的程序计数器,内存指针和上下文数据。大致如图

组织进程
可以在内核源代码⾥找到它。所有运⾏在系统⾥的进程都以task_struct链表的形式存在内核⾥。

4.查看进程

4.1. 进程的信息可以通过 /proc 系统⽂件夹查看

如:要获取PID为1的进程信息,你需要查看 /proc/1 这个⽂件夹。

值得注意的是,在这个对应pid的文件夹下我们可以看到,记录了 当前进程的二进制可执行代码的路径,和当前进程的工作路径。这个工作路径可以在 Linux 下,可以使用 C 语言中的chdir函数来修改进程的工作路径。

cpp 复制代码
#include <unistd.h>

int chdir(const char *path);

4.2. ⼤多数进程信息同样可以使⽤top和ps这些⽤⼾级⼯具来获取

5.通过系统调⽤获取进程标⽰符

• 进程id(PID)
• ⽗进程id(PPID)

cpp 复制代码
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
    printf("pid: %d\n", getpid());
    printf("ppid: %d\n", getppid());
    return 0;
}

6.通过系统调⽤创建进程-fork初识

在Linux下,进程的增加的方式是由父进程创建子进程。
在多次运行上面代码时,我们会发现,pid一直在变化,ppid却一直没变。
其实这里是因为,我们的进程是bash的子进程。
运⾏ man fork

fork有两个返回值
⽗⼦进程代码共享,数据各⾃开辟空间,私有⼀份(采⽤写时拷⻉)
fork 之后通常要⽤ if 进⾏分流

cpp 复制代码
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
    int ret = fork();
    if(ret < 0){
        perror("fork");
        return 1;
    }
    else if(ret == 0){ //child
        printf("I am child : %d!, ret: %d\n", getpid(), ret);
    }else{ //father
        printf("I am father : %d!, ret: %d\n", getpid(), ret);
    }
    sleep(1);
    return 0;
}

fork为什么会有两个返回值?

  • 本质:fork() 的核心作用是复制当前进程(父进程),创建一个几乎完全相同的新进程(子进程)。这两个进程从 fork() 调用的下一行代码开始并发执行。
  • 返回值的作用:为了让父进程和子进程能够区分自己的身份,fork() 在两个进程中返回不同的值:
    • 在父进程中:返回子进程的进程 ID(PID,一个正整数)。
    • 在子进程中:返回 0
    • 若出错:父进程返回 -1,且不会创建子进程。

两个返回值各种给⽗⼦如何返回?

  • 执行流程:
    1. 父进程调用 fork(),内核创建子进程。
    2. 父进程继续执行,fork() 返回子进程的 PID(正整数)。
    3. 子进程从 fork() 的返回处开始执行,fork() 返回 0

为什么设计这样的返回值?

  • 子进程通过 0 确认身份:子进程可以直接通过 if (pid == 0) 判断自己是子进程。
  • 父进程需要子进程的 PID:父进程通常需要管理子进程(如等待子进程结束、发送信号),因此返回子进程的 PID 便于操作
相关推荐
来鸟 鸣间21 分钟前
vscode 连接远程服务器
服务器·ide·vscode
伤不起bb39 分钟前
Kafka 消息队列
linux·运维·分布式·kafka
Hello.Reader1 小时前
Git 安装全攻略Linux、macOS、Windows 与源码编译
linux·git·macos
龙仔7251 小时前
华为云CentOS配置在线yum源,连接公网后,逐步复制粘贴,看好自己对应的版本即可,【新手必看】
linux·centos·华为云
tiging1 小时前
centos实现SSH远程登录
linux·centos·ssh
Wooden-Flute1 小时前
五、查询处理和查询优化
服务器·数据库·oracle
好多知识都想学2 小时前
Linux 文件处理器 sed 和 awk 详细讲解
linux·运维·ubuntu
Johny_Zhao2 小时前
阿里云数据库Inventory Hint技术分析
linux·mysql·信息安全·云计算·系统运维
FBI HackerHarry浩2 小时前
云计算 Linux Rocky day05【rpm、yum、history、date、du、zip、ln】
linux·运维·云计算·腾讯云
敲键盘的小夜猫2 小时前
RunnablePassthrough介绍和透传参数实战
java·服务器·前端