【计算机网络】fork()+exec()创建新进程(僵尸进程及孤儿进程)

article/2025/7/23 0:51:29

文章目录

  • 一、基本概念
    • 1. fork() 系统调用
    • 2. exec() 系列函数
  • 二、典型使用场景
    • 1. 创建子进程执行新程序
    • 2. 父子进程执行不同代码
  • 三、核心区别与注意事项
  • 四、组合使用技巧
    • 1. 重定向子进程的输入/输出
    • 2. 创建多级子进程
  • 五、常见问题与解决方案
    • 僵尸进程(Zombie Process)
    • 孤儿进程(Orphan Process)
  • 六、总结

在UNIX/Linux系统中,fork() 和 exec() 是创建和运行新进程的核心系统调用。以下是它们的详细解释和典型用法。

一、基本概念

1. fork() 系统调用

功能:创建当前进程的一个子进程,子进程是父进程的几乎完全复制(包括内存空间、文件描述符等)。
返回值:
父进程:返回子进程的进程ID(PID,正整数)。
子进程:返回0。
错误:返回-1(如内存不足或进程数达到上限)。

2. exec() 系列函数

功能:用新程序替换当前进程的内存空间、代码段和数据段,执行新程序。
常见函数:

execl(const char *path, const char *arg, ...);
execv(const char *path, char *const argv[]);
execle(const char *path, const char *arg, ..., char *const envp[]);
execve(const char *path, char *const argv[], char *const envp[]);
execlp(const char *file, const char *arg, ...);
execvp(const char *file, char *const argv[]);

区别:l 表示参数列表(list),v 表示参数数组(vector),p 表示自动搜索PATH路径,e 表示自定义环境变量。

二、典型使用场景

1. 创建子进程执行新程序

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>int main() {pid_t pid = fork();  // 创建子进程if (pid < 0) {       // 错误处理perror("fork failed");exit(EXIT_FAILURE);} else if (pid == 0) {  // 子进程// 执行新程序(如/bin/ls)execl("/bin/ls", "ls", "-l", NULL);// 如果exec返回,说明执行失败perror("exec failed");exit(EXIT_FAILURE);} else {  // 父进程int status;waitpid(pid, &status, 0);  // 等待子进程结束printf("Child process exited with status %d\n", WEXITSTATUS(status));}return 0;
}

执行流程:

父进程调用 fork() 创建子进程。
子进程调用 execl() 加载 /bin/ls 程序,替换自身代码。
父进程通过 waitpid() 等待子进程结束,并获取退出状态。

2. 父子进程执行不同代码

#include <stdio.h>
#include <unistd.h>int main() {pid_t pid = fork();if (pid < 0) {perror("fork failed");} else if (pid == 0) {  // 子进程printf("Child process (PID=%d)\n", getpid());// 子进程逻辑...} else {  // 父进程printf("Parent process (PID=%d, Child PID=%d)\n", getpid(), pid);// 父进程逻辑...}return 0;
}

关键点: 父子进程通过 fork() 的返回值区分执行逻辑。 子进程继承父进程的文件描述符和内存状态(但后续修改互不影响)。

三、核心区别与注意事项

在这里插入图片描述

四、组合使用技巧

1. 重定向子进程的输入/输出

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>int main() {pid_t pid = fork();if (pid == 0) {  // 子进程// 重定向标准输出到文件int fd = open("output.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644);dup2(fd, STDOUT_FILENO);  // 将标准输出(1)替换为文件描述符fdclose(fd);// 执行ls命令,输出会写入文件而非终端execl("/bin/ls", "ls", "-l", NULL);perror("exec failed");} else {wait(NULL);  // 等待子进程结束printf("Output redirected to output.txt\n");}return 0;
}

2. 创建多级子进程

#include <stdio.h>
#include <unistd.h>int main() {pid_t pid1 = fork();if (pid1 == 0) {  // 一级子进程printf("First child (PID=%d)\n", getpid());pid_t pid2 = fork();if (pid2 == 0) {  // 二级子进程printf("Second child (PID=%d, Parent=%d)\n", getpid(), getppid());} else {wait(NULL);  // 一级子进程等待二级子进程}} else {wait(NULL);  // 父进程等待一级子进程}return 0;
}

五、常见问题与解决方案

僵尸进程(Zombie Process)

问题:子进程结束后,父进程未调用 wait() 获取其退出状态,导致子进程残留进程表项。
解决方案:
父进程通过 wait() 或 waitpid() 回收子进程资源。
信号处理:捕获 SIGCHLD 信号并在处理函数中调用 wait()。

孤儿进程(Orphan Process)

问题:父进程先于子进程结束,子进程被init进程(PID=1)收养。
影响:孤儿进程仍可正常运行,但可能导致资源管理复杂化。
exec() 失败处理
原因:路径错误、权限不足、文件损坏等。
检查方法:调用 exec() 后立即检查返回值(若返回则表示失败),并通过 errno 获取具体错误码。

六、总结

fork() 用于创建新进程,父子进程并行执行相同代码(通过返回值区分逻辑)。
exec() 用于加载并执行新程序,替换当前进程的内存空间。
组合使用:先 fork() 创建子进程,再在子进程中 exec() 执行新程序,实现多任务处理。
理解这两个系统调用是掌握UNIX/Linux进程管理的基础,在开发多进程应用(如服务器、shell脚本解释器)时尤为重要。


http://www.hkcw.cn/article/gfjhrVeGSX.shtml

相关文章

Selenium操作指南(全)

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 大家好&#xff0c;今天带大家一起系统的学习下模拟浏览器运行库Selenium&#xff0c;它是一个用于Web自动化测试及爬虫应用的重要工具。 Selenium测试直接运行在…

Linux研学-入门命令

一 目录介绍 1 介绍 Linux与Windows在目录结构组织上差异显著&#xff1a;Linux采用树型目录结构&#xff0c;以单一根目录/为起点&#xff0c;所有文件和子目录由此向下延伸形成层级体系&#xff0c;功能明确的目录各司其职&#xff0c;使文件系统层次清晰、逻辑连贯&#xf…

GSR 手环能耗数据实测:STM32 与 SD NAND 的功耗优化成果

文章目录 卓越性能强化安全高效能效图形处理优势丰富集成特性模拟模块实时监控保障数据完整性提升安全性与可靠性测量原理采样率相关 在智能皮电手环及数据存储技术不断迭代的当下&#xff0c;主控 MCU STM32H750 与存储 SD NAND MKDV4GIL-AST 的强强联合&#xff0c;正引领行业…

PCIe-PCI、PCIe中断机制概述

PCI、PCIe中断概述 PCIe 中断机制在继承 PCI 传统中断&#xff08;INTx&#xff09;的基础上&#xff0c;引入了更高效的 MSI/MSI-X 方案&#xff0c;以提升设备性能并减少 CPU 轮询开销。以下是核心要点及技术演进&#xff1a; ⚙️ ​​一、PCIe 中断类型与演进​​ ​​IN…

知识图谱增强的大型语言模型编辑

https://arxiv.org/pdf/2402.13593 摘要 大型语言模型&#xff08;LLM&#xff09;是推进自然语言处理&#xff08;NLP&#xff09;任务的关键&#xff0c;但其效率受到不准确和过时知识的阻碍。模型编辑是解决这些挑战的一个有前途的解决方案。然而&#xff0c;现有的编辑方法…

建立连接后 TCP 请求卡住

大家读完觉得有意义记得关注和点赞&#xff01;&#xff01;&#xff01; 这篇文章描述了一个内核和BPF网络问题 以及故障排除步骤&#xff0c;这是一个值得深入研究的有趣案例 Linux 内核网络复杂性。 目录 1 故障报告 1.1 现象&#xff1a;概率健康检查失败 1.2 范围&am…

C++核心编程_赋值运算符重载

4.5.4 赋值运算符重载 c编译器至少给一个类添加4个函数 默认构造函数(无参&#xff0c;函数体为空) 默认析构函数(无参&#xff0c;函数体为空) 默认拷贝构造函数&#xff0c;对属性进行值拷贝 赋值运算符 operator, 对属性进行值拷贝 如果类中有属性指向堆区&#xff0c;做…

深度学习笔记25-RNN心脏病预测(Pytorch)

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 一、前期准备 1.数据处理 import torch.nn.functional as F import numpy as np import pandas as pd import torch from torch import nn dfpd.read_csv(r&…

基于 HT for Web 轻量化 3D 数字孪生数据中心解决方案

一、技术架构&#xff1a;HT for Web 的核心能力 图扑软件自主研发的 HT for Web 是基于 HTML5 的 2D/3D 可视化引擎&#xff0c;核心技术特性包括&#xff1a; 跨平台渲染&#xff1a;采用 WebGL 技术&#xff0c;支持 PC、移动端浏览器直接访问&#xff0c;兼容主流操作系统…

CIO大会, AI课笔记手稿分享

能认真听课的CIO不多了 能认真听课的CIO还能记笔记的不多了 能认真听课的CIO还能记笔记的字写得好的不多了

stl三角面元文件转颗粒VTK文件

效果展示&#xff1a; import os import sys import json import argparse import numpy as np import pandas as pd import open3d as o3d from glob import globPARTICLE_RADIUS 0.025def stl_to_particles(objpath, radiusNone):if radius is None:radius PARTICLE_RADIU…

vue为什么点击两遍才把参数传递过去

先说一下场景&#xff0c;就是我把云服务器这个下拉选择框分别初始化之后&#xff0c;然后点击新建权限然后就打开了右侧的抽屉式的对话框&#xff0c;页面上那个文字信息是传递过来了。那个是正确的&#xff0c;但是我请求接口的时候&#xff0c;发现请求的接口的参数总是要慢…

NodeMediaEdge通道管理

NodeMediaEdge任务管理 简介 NodeMediaEdge是一款部署在监控摄像机网络前端中&#xff0c;拉取Onvif或者rtsp/rtmp/http视频流并使用rtmp/kmp推送到公网流媒体服务器的工具。 在未使用NodeMediaServer的情况下&#xff0c;或者对部分视频流需要单独推送的需求&#xff0c;也可…

2025年- H59-Lc167--207.课程表(拓扑排序、BFS)-需二刷--Java版

1.题目描述 2.思路 记录每门课程的前置课程数量&#xff0c;记录每门课程是哪些课程的前置课程。 &#xff08;1&#xff09;如果有向图中的拓扑图中存在环&#xff0c;则说明所有的课程是无法完成的。 &#xff08;2&#xff09;使用拓扑排序&#xff0c;在图中每个节点的入度…

MQTT的Thingsboards的使用

访问云服务 https://thingsboard.cloud/ 新建一个设备 弹出 默认是mosquittor的客户端。 curl -v -X POST http://thingsboard.cloud/api/v1/tnPrO76AxF3TAyOblf9x/telemetry --header Content-Type:application/json --data "{temperature:25}" 换成MQTTX的客户…

代码随想录算法训练营第60期第五十二天打卡

大家好&#xff0c;昨天我们重点讲解了单调栈的问题&#xff0c;我们今天的题目还是继续围绕单调栈展开&#xff0c;我们上节课其实对单调栈已经有了大致的了解&#xff0c;今天的第一道题目大家务必要注意很重要&#xff0c;接雨水问题我们会涉及到单调栈与双指针&#xff0c;…

新能源集群划分+电压调节!基于分布式能源集群划分的电压调节策略!

适用平台&#xff1a;MatlabYalmip Cplex (具体操作已在程序文件中说明) 参考文献&#xff1a;基于分布式能源集群化分的电压调节策略[D]. 一、文献解读 1. 主要内容/创新点 提出了一种基于分布式能源集群化的电压调节策略&#xff0c;计及分布式能源的有功、无功调节能力&a…

【C++】22. 红黑树封装实现Mymap和Myset

上一章节我们实现了红黑树&#xff0c;这一章节我们就用红黑树封装来实现一个我们自己的map和set 1. 源码及框架分析 SGI-STL 3.0版本的源代码中&#xff0c;map和set的实现主要分布在若干头文件中&#xff0c;这些头文件构成了这两个容器的完整实现架构&#xff1a; 核心头文…

论文速读《UAV-Flow Colosseo: 自然语言控制无人机系统》

论文链接&#xff1a;https://arxiv.org/abs/2505.15725项目主页&#xff1a;https://prince687028.github.io/UAV-Flow/ 0. 简介 近年来&#xff0c;无人机技术蓬勃发展&#xff0c;但如何让无人机像智能助手一样理解并执行人类语言指令&#xff0c;仍是一个前沿挑战。现有研…

关于表连接

目录 1.左连接 2.右连接 3.内连接 4.全外连接 5.笛卡尔积 -- 创建表A CREATE TABLE A(PNO VARCHAR2(10) PRIMARY KEY, PAMT NUMBER, A_DATE DATE);-- 向表A插入数据 INSERT INTO A VALUES (01001, 100, TO_DATE(2005-01-01, YYYY-MM-DD)); INSERT INTO A VALUES (010…