Linux——进程控制:创建、终止、等待、替换

news2025/1/11 2:32:37

进程创建

fork

#include <unistd.h>
pid_t fork(void);

操作系统做了什么?

调用fork之后,内核的工作:

  1. 分配新的内存块和内核数据结构给子进程
  2. 将父进程部分数据结构内容拷贝至子进程
  3. 添加子进程到系统进程列表当中
  4. fork返回,开始调度器调度

进程 = 内核数据结构 + 代码和数据

创建子进程的过程就是:创建子进程的内核数据结构,即task_struct + mm_struct + paper_table等等,继承父进程的代码,同时拷贝数据。

fork之后,操作系统会create一个新的process,copy_process,copy_mem,把父进程的所有资源(包括代码段,数据段、堆栈数据等),以及寄存器的值继承给子进程。之后再以写时拷贝的形式保证进程的数据独立性。

do_fork

int sys_fork(struct pt_regs *regs)
{
    return do_fork(SIGCHLD, regs->sp, regs, 0, NULL, NULL);
}

asmlinkage int sparc_do_fork(unsigned long clone_flags,
                             unsigned long stack_start,
                             struct pt_regs *regs,
                             unsigned long stack_size)
{
    unsigned long parent_tid_ptr, child_tid_ptr;
    unsigned long orig_i1 = regs->u_regs[UREG_I1];
    long ret;
    parent_tid_ptr = regs->u_regs[UREG_I2];
    child_tid_ptr = regs->u_regs[UREG_I4];
    ret = do_fork(clone_flags, stack_start,
              regs, stack_size,
              (int __user *) parent_tid_ptr,
              (int __user *) child_tid_ptr);
    /* If we get an error and potentially restart the system
     * call, we're screwed because copy_thread() clobbered
     * the parent's %o1.  So detect that case and restore it
     * here.
     */
    if ((unsigned long)ret >= -ERESTART_RESTARTBLOCK)
        regs->u_regs[UREG_I1] = orig_i1;
    return ret;
}

fork返回之后,两个进程的PC指针都指向fork函数之后的代码,但实际上是父子进程共享了整个代码。子进程运行的时候是从pc指针中取值得到下一次运行的代码的地址。
当然,子进程完全可以通过goto等方法让执行流回到fork之前的地方。

一个有趣的现象

#include <stdio.h>
#include <unistd.h>
#include <cstdlib>

int main()
{
  printf("我是父进程,我要来执行fork了\n");
  pid_t id = fork();

again:  printf("这是fork之后的代码,子进程跳转后才能访问到的\n");

  if(id == 0)
  {
    printf("我是子进程 pid = %d, ppid = %d\n",getpid(), getppid());
    for(int i = 0; i < 3; i++)
      sleep(1);
      
    printf("子进程要跳转了\n");
    goto again;
    exit(1);
  }
  while(1)
  {
    printf("我是父进程 pid = %d, ppid = %d\n",getpid(), getppid());
    for(int i = 0; i < 3; i++)
    {
      sleep(1);
    }
    break;
  }
  return 0;
}


Ctrl+C杀掉进程后,命令行开始提供服务,过了一会还会继续打印?这里看起来是Ctrl+C不起作用
这不难理解,./a.out之后,运行的是从始至终的父进程,父进程会被杀掉,但是子进程还在一直运行。父进程挂掉之后,shell继续提供服务,但是子进程还在向显示器打印。要用kill命令发送信号杀掉子进程,子进程会进入终止态,但是因为父进程早就挂掉了,所以子进程会被init领养,之后被回收资源。


进程终止

进程退出

进程退出的场景:运行完毕,结果正确;运行完毕,结果不正确;运行没完毕,被终止。
echo $?命令可以查询最近一次进程执行完毕时对应进程的退出码,退出码是一个8位无符号数。

main函数为什么要return,return给谁,为什么是0?
为了向操作系统或者其他程序报告自己的运行状态。
函数的返回值为0表示成功,因为成功只有一个标志就行,不需要知道原因,失败可以有无数个原因。
例如STM32开发中调用MPU6050的DMP库的初始化函数,如果不成功会有多种错误原因,用正整数可以标识。

如何终止进程?

#include <unistd.h>
void _exit(int status);
void exit(int status);

正常终止

  1. 在main函数中return
  2. 用exit或者_exit函数
    区别:exit会终止进程且刷新缓冲区,但是_exit不会。
    调用exit,系统会关闭所有打开的流,刷新缓冲区,之后再调用_exit。

虽然status是int,但是操作系统会把int转换成uint,因为退出码的范围是0-255。所以exit(-1)时,echo $? 结果是255。

异常终止

如收到信号。


进程等待

为什么要等待?

如果子进程退出,父进程不回收就会变成僵尸进程,操作系统是无法杀死的,可能内存泄漏,除非父进程也结束了,子进程被操作系统领养。因此父进程应该等待子进程,回收子进程的资源,获取其退出信息,例如退出码。

如何进程等待?

#include <sys/types.h>
#include <sys/wait.h>
pid_t wait(int* status);
pid_t waitpid(pid_t pid, int* status, int options);

调用这两个函数,可以从task_struct中拿到进程的退出状态,退出状态是进程退出后仍留在数据结构中的。

wait

  • wait的参数是输出型参数,等待任意一个子进程,并输出退出信息。

waitpid

  • pid:-1表示任意进程,大于0的数表示要等待的进程的pid。
  • status:该进程的退出状态。
  • options:0表示阻塞等待、WNOHANG表示非阻塞等待。
  • 返回值:大于0表示等待成功,并且进程已退出;等于0表示等待成功,但是进程还没有退出。

进程阻塞

调用scanf和cin的时候,没有输入就会一直等待,task_struct的R状态变为S状态,从运行队列转移到等待队列。

非阻塞等待

等待的时候如果事件没就绪就直接返回,执行别的东西,待会再来检查。多次调用费阻塞等待接口的检测行为称为轮询检测

等待的结果

退出码的组成

status的低八位是进程的退出信号,高8位是进程的退出码。当进程被杀死了还会有一个core_dump标志位。

printf("退出信号 = %d, 退出码 = %d\n", (status & 0x7F), (status >> 8) & 0x7F);

例如:当用kill杀死进程的时候,比如kill -9 进程, 进程的低八位就会收到9号信号。
如果一个进程收到信号,就说明进程出现异常,这时候退出码就没用了,因为不是正常退出,所以只关心收到的信号。


进程替换

什么是进程替换?

之前创建进程后,父子进程共享代码。如果想让子进程执行其他逻辑呢?

  1. 进程替换不会创建新进程,因为进程替换只是将该进程的数据替换为指定的可执行程序。而进程PCB没有改变,所以不是新的进程,pid不变。
  2. 进程替换后,如果替换成功则开始执行新程序,原替换函数之后的代码不会执行,因为进程替换是覆盖式的替换,替换成功后进程原来的代码就消失了。如果进程替换失败则会执行原替换函数后的代码。

原理

将磁盘中的别的程序加载到内存中,通过写时拷贝,重新建立页表的映射,将原子进程中数据段和代码段的映射改变为到别的程序的映射,这样父子进程的代码和数据就完全分离了。

进程替换的接口

#include <unistd.h>
int execl (const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ..., char * const envp[]);

int execv (const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);

execl

  • path:用来替换的程序路径:/usr/bin/ls,也可以是相对路径。要执行一个程序,系统需要知道如何执行这个程序,就要带参数:ls -a -l
  • arg:表示可变参数,用于传入选项。以NULL为终止符表示参数传递完毕。
  • 返回值不用判断,因为替换成功不会返回,替换失败将会执行原文件后续代码。
  • 调用方式:execl ("/usr/bin/ls", "ls", "-a", "-l", NULL);

execlp

(p代表环境变量path)

  • file:要替换的程序名,可以是相对路径,也可以在环境变量中搜索。
  • arg:可变参数,跟上面一样。
  • 返回值:不用判断
  • 调用方式:execlp(“./hello”,“./hello” ,NULL);

execv和execvp

跟execl的区别是可变参数变成字符串数组。

char* commands[] = {"./hello", NULL};                                        
std::cout << "我要替换了\n";
execv("./hello",commands);

execle和execve

则是可以自己导入环境变量。

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

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

相关文章

一个简单案例理解为什么在多线程的应用中要使用锁

需求:使用10个线程,同时对一个值count进行加一操作,每个线程对count加100000次,最终使得count1000000 第一版代码:不加锁 ​​​lock.c #include<stdio.h> #include<pthread.h>#define THREAD_COUNT 10void *thread_callback(void *arg){int *pcount(int*)arg;in…

计算机网络--网络传输基本概念

什么是IP地址&#xff1f; 在计算机出厂的时候&#xff0c;有一个唯一标识的物理地址。但是因为厂商不同等各种原因&#xff0c;用来标识一台计算机在网络中是比较麻烦的&#xff0c;于是出现了IP地址&#xff0c;IP地址是互联网协议地址的意思&#xff0c;是“Internet Protoc…

【Matlab】数字图像的 SVD 分解

奇异值分解 (SVD, Singular Value Decomposition) 是线性代数中一种重要的矩阵变换方法&#xff0c;对矩阵进行 SVD 分解&#xff0c;可以把复杂的矩阵简化&#xff0c;从而提取出重要的信息。数字图像的 SVD 分解是对数字图像建模的一种方法与工具&#xff0c;可以应用于图像压…

操作系统01-导论

一、概述 操作系统&#xff1a;英文是operating system&#xff0c;OS 它的作用运行用户程序&#xff08;核心目标&#xff09;高效使用计算机&#xff08;面向系统&#xff09;方便使用计算机&#xff08;面向用户&#xff09; 二、内容 2.1 现代计算机系统 一个或多个CPU和…

机器学习:self supervised learning

340M 参数 BERT 自监督学习的目标跟目标越接近越好。 一个任务&#xff1a;预测句子中被mask的词&#xff1a; BERT通过Masking Input来获得训练数据 mask有两种做法&#xff1a; 将某个字token换成一个特殊符号&#xff08;代表盖住&#xff09; 随机把某个字换成另外一个…

chatgpt赋能python:Python程序怎么打包

Python程序怎么打包 Python作为一种功能强大的编程语言&#xff0c;它的很多应用都需要打包成可执行文件或者可以方便部署的代码。本篇文章将介绍Python程序打包的方法及步骤。 为什么需要打包&#xff1f; 分享代码或程序&#xff1a;当你编写了一个Python程序并且想要分享给…

【⑥MySQL多表查询】:让你的数据检索更高效

前言 ✨欢迎来到小K的MySQL专栏&#xff0c;本节将为大家带来MySQL中多表查询相关知识的讲解 目录 前言一、多表关系二、多表查询1、交叉连接2、内连接3、外连接 三、集合运算四、七种JOINS实现五、多表查询练习六、总结 一、多表关系 ✨项目开发中&#xff0c;在进行数据库表结…

MySQL生产环境高可用架构详解

一、MySQL高可用集群介绍 1、数据库主从架构与分库分表 随着现在互联网的应用越来越大&#xff0c;数据库会频繁的成为整个应用的性能瓶颈。而 我们经常使用的MySQL数据库&#xff0c;也会不断面临数据量太大、数据访问太频繁、数据 读写速度太快等一系列的问题。所以&#xf…

记录分享在10年老的商务本Dell E6230上安装Debian 12的过程,遇到的问题和解决方法

原先在笔记本上安装的是Debian 9&#xff0c;最近发现无法更新了&#xff0c;查一下发现&#xff0c;所有的“源”只支持deb10&#xff0c;11 和 12&#xff0c;所以特意订了一块新的硬盘来安装新系统&#xff0c;前后倒腾了两天多。 在此记录这个过程中遇到的问题和解决的方法…

Mysql主从复制和读写分离(期望日子清静,抬头皆是温柔)

文章目录 一、读写分离1.什么是读写分离?2.为什么要读写分离呢?3.什么时候要读写分离?4.读写分离原理5.读写分离方式&#xff08;1&#xff09;基于程序代码内部实现&#xff08;2&#xff09;基于中间代理层实现 二、主从复制1.主从复制与读写分离的关系2.mysql支持的复制类…

langchain源码阅读系列(一)之LLM输入输出管理

原文首发于博客文章OpenAI 文档解读 LangChain 主体分为 6 个模块&#xff0c;分别是对&#xff08;大语言&#xff09;模型输入输出的管理、外部数据接入、链的概念、&#xff08;上下文记忆&#xff09;存储管理、智能代理以及回调系统&#xff0c;通过文档的组织结构&#x…

如何编写一个最简单的 udp 版本的 echo server 和 echo client(小白也懂!)

目录 目的 第一步 编写Server(服务器) 第二步 创建Server的各类参数 第三步 实现具体的Server内容 第四步 编写Client(客户端) 实现具体的Client内容 总流程 总代码 源码下载 目的 我们编写一个udp 版本的 echo server 和 echo client 实现在自己电脑上通过客户端…

open【部署、使用教程】

目录 【1】创建证书 【2】安装openVPN-Server端并配置 【3】将证书移动到相对路径 【4】开启内核转发功能&#xff0c;否则会无法启动openVPN 【5】启动服务&#xff0c;加入开机自启 【6】启动后服务端会生成一个tun0的虚拟网卡&#xff0c;用于不同网段之间相互通信 【…

VUE2.0集成 Markdown 编辑器

Markdown编辑器的使用 这是一款基于Vue的markdown编辑器。既可以用来编辑Markdown语法&#xff0c;又可以用来解析 效果图,mavonEditor实现了Markdown集成 Markdown是一种标记语言&#xff0c;相较于word文档更加清晰方便&#xff0c;适合进行笔记等。将Markdown集成进入自己项…

Matlab使用S函数

什么是S函数&#xff1f; S-函数是系统函数&#xff08;System Function&#xff09;的简称&#xff0c;在 Simulink 中用非图形化的方式来描述一个模块。一个完整的S-函数结构体系包含了描述一个动态系统所需要的全部能力。使用S-函数用户可以向 Simulink 模型中添加自己的模块…

【PCB专题】Allegro中设置泪滴

PCB绘制完成后有时按需要对PCB进行添加泪滴的操作是非常必要的。 添加泪滴的作用主要是: 信号传输时平滑阻抗,减少阻抗的急剧跳变,避免高频信号传输时由于线宽突然变小而造成反射。 焊接时可以保护焊盘,避免多次焊接时焊盘的脱落,生产时可以避免蚀刻不均,以及过孔偏位出…

一键安装和导出当前Python项目的依赖包总结

创建python环境&#xff0c;配置一个python运行项目。在项目可以运行的环境下&#xff0c;导出该项目所依赖包到一个requirements.txt文档中。在另一个纯净环境中&#xff0c;快速批量安装项目所依赖的包&#xff0c;便于快速进行项目迁移 一、导出当前Python项目的依赖包1、方…

【数据结构与算法】3、虚拟头节点、动态数组的缩容、动态数组和单链表的复杂度、数组的随机访问

目录 一、虚拟头节点二、数组的随机访问三、动态数组、链表复杂度分析四、动态数组 add(E element) 复杂度分析五、动态数组的缩容 一、虚拟头节点 &#x1f33c; 为了让代码更加精简&#xff0c;统一所有节点的处理逻辑&#xff0c;可以在最前面增加一个虚拟的头节点&#xf…

2023 年最佳 C++ IDE

文章目录 前言1. Visual Studio2. Code::Blocks3. CLion4. Eclipse CDT&#xff08;C/C 开发工具&#xff09;5. CodeLite6. Apache NetBeans7. Qt Creator8. Dev C9. C Builder10. Xcode11. GNAT Programming Studio12. Kite总结 前言 要跟踪极佳 IDE&#xff08;集成开发环境…

嵌入式系统复习要点

目录 1、嵌入式系统的核心部分主要由硬件和软件两部分组成&#xff1a; 2、嵌入式系统硬件&#xff1a; 3、嵌入式处理器从体系上分类&#xff0c;可以分为冯诺依曼结构和哈佛结构两种&#xff1a; 4、几类常见的嵌入式处理器类型&#xff1a; 5、MCU组成结构&#xff1a;…