news 2026/6/9 20:57:23

进程创建-fork和system函数使用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
进程创建-fork和system函数使用

进程创建-fork和system函数使用

文章目录

  • 进程创建-fork和system函数使用
  • 1、system()函数
  • 2、fork()函数
  • 3、继承关系:
  • 4、一些FAQ

一般情况下我们可以打开终端,直接执行./demo等命令执行一个程序,此时程序以进程的形式运行,大概率程序的父进程是执行此命令对应的shell进程。

实际应用中大概是在程序中完成进程的创建,简单常用的可以用system命令调用,比如system(“./demo &”) 或者使用fork创建一个子进程,本文介绍这两种方法的区别和使用注意事项。

1、system()函数

主程序在代码中执行system函数相当于利用shell新开一个独立的进程,此时shell的父进程为原始程序,system中执行的命令是新开一个进程,此时该命令的父进程为shell,相对主程序来说,system中执行的进程变成了孙子进程,如下:

主进程(PID=1000)
└── sh(PID=1001) # system启动的shell
└── sleep(PID=1002) # shell执行的命令

另外需要注意的是,sytem中使用&和不使用&存在区别,不使用&会阻塞主程序,直到system调用的程序运行结束才会执行主程序的代码。如果system调用的程序是一直运行的,需要加&符号后台运行,此时system会立即返回,不会阻塞主进程,但由于system返回了、会导致后台进程成为孤儿进程。

#include <stdlib.h> #include <stdio.h> int main() { printf("父进程开始\n"); // 情况1: 阻塞执行 printf("=== 阻塞执行 ===\n"); int ret1 = system("sleep 3 && echo '阻塞命令完成'"); printf("system返回: %d\n", ret1); // 情况2: 非阻塞执行(使用&) printf("\n=== 非阻塞执行 ===\n"); int ret2 = system("sleep 3 && echo '非阻塞命令完成' &"); printf("system立即返回: %d\n", ret2); printf("父进程继续执行...\n"); sleep(5); // 等待后台进程 return 0; }

2、fork()函数

fork函数是在主程序运行过程中直接创建子进程,需要关注的是fork函数是一次调用,两次返回,fork()的返回值有三种情况,它用来区分父进程和子进程,这种设计使得父进程和子进程可以在同一个代码中执行不同的逻辑路径,具体先执行父进程还是子进程受到操作系统调度影响,是非阻塞的

#include <unistd.h> #include <stdio.h> #include <sys/types.h> int main() { printf("准备调用fork()...\n"); pid_t pid = fork(); if (pid < 0) { fprintf(stderr, "fork失败\n"); return 1; } else if (pid == 0) { // 子进程 printf("这是子进程:\n"); printf(" fork()返回值: %d\n", pid); // 应该是0 printf(" 自己的PID: %d\n", getpid()); printf(" 父进程PID: %d\n", getppid()); // 子进程可以执行不同的任务 sleep(1); // 模拟工作 printf("子进程结束\n"); return 10; // 子进程退出码 } else { // 父进程 printf("这是父进程:\n"); printf(" fork()返回值: %d (这是子进程的PID)\n", pid); printf(" 自己的PID: %d\n", getpid()); printf("父进程结束\n"); } return 0; }

程序的返回为:

准备调用fork()... 这是父进程: fork()返回值: 3683 (这是子进程的PID) 自己的PID: 3682 父进程结束 这是子进程: fork()返回值: 0 自己的PID: 3683 父进程PID: 3682 图解说明: 调用 fork() ↓ ┌───────────────┐ │ 创建子进程副本 │ └───────────────┘ ↓ 父进程继续执行 子进程开始执行 ↓ ↓ 返回子进程的PID 返回 0 ↓ ↓ 执行 pid > 0 分支 执行 pid == 0 分支 ↓ ↓ 各自独立运行,互不影响

fork函数在创建进程之后可以配合exec()的各种函数,即在返回的pid=0的分支调用exec函数执行自己新开的程序命令。

#include <unistd.h> // 函数原型 int execl(const char *path, const char *arg, ... /* (char *) NULL */); int execlp(const char *file, const char *arg, ... /* (char *) NULL */); int execle(const char *path, const char *arg, ... /*, (char *) NULL, char * const envp[] */); int execv(const char *path, char *const argv[]); int execvp(const char *file, char *const argv[]); int execvpe(const char *file, char *const argv[], char *const envp[]); l (list):参数以列表形式传递 v (vector):参数以数组形式传递 p (PATH):在PATH环境变量中查找程序 e (environment):可以指定新的环境变量

具体的函数用法可以网上搜,举两个常用简单例子:

//// execl pid_t pid = fork(); if (pid == 0) { // 子进程 execl("./codebin", "codebin", "code.cfg", NULL); // 第一个参数需要时=为程序的执行路径 // 如果exec失败,才会执行到这里 perror("execl失败"); exit(1); } //// execv pid_t pid = fork(); if (pid == 0) { // 子进程 char *args[] = {"./codebin", "code.cfg", NULL}; execv("./codebin", args); perror("execv失败"); exit(1);

3、继承关系:

system和fork在创建进程的时候会继承父进程的很多属性,例如:

1.环境变量:当前进程的所有环境变量都会被继承
2.当前工作目录:子进程会继承父进程的工作目录
3.进程组和会话:通常在同一会话和进程组中
4.信号处理:某些信号处理方式会继承
5.文件描述符:打开的文件描述符会继承(标准输入/输出/错误除外,会被重定向)等

demo例子:

#include <unistd.h> #include <sys/types.h> #include <sys/wait.h> #include <stdio.h> #include <stdlib.h> #include <fcntl.h> int main() { // 环境变量 - 会被继承 setenv("MY_VAR", "parent_value", 1); // 文件描述符 - 会被继承 int fd = open("test.txt", O_CREAT | O_WRONLY, 0644); write(fd, "来自父进程\n", 12); // 进程组、会话、工作目录等 - 会被继承 printf("父进程PID: %d, 工作目录: %s\n", getpid(), getcwd(NULL, 0)); pid_t pid = fork(); if (pid == 0) { // 子进程 printf("子进程PID: %d\n", getpid()); printf("环境变量MY_VAR: %s\n", getenv("MY_VAR")); // 可以访问父进程打开的文件 write(fd, "来自子进程\n", 12); close(fd); // 改变环境变量(只影响子进程) setenv("MY_VAR", "child_value", 1); // 执行新程序 char *argv[] = {"./other_program", NULL}; char *envp[] = {"MY_VAR=exec_value", "PATH=/bin", NULL}; // 不同的exec变体提供不同的控制 // execv("./other_program", argv); // 继承所有环境变量 // execve("./other_program", argv, envp); // 指定新环境变量 // execl("/bin/ls", "ls", "-l", NULL); // 列表参数 exit(0); } else { // 父进程的环境变量不变 printf("父进程中的MY_VAR: %s\n", getenv("MY_VAR")); close(fd); wait(NULL); } return 0; }

4、一些FAQ

继承有时候可能存在问题,比如子进程如果一直占用某些文件描述符,即使父继承显示关闭了,需要卸载驱动或者其它不允许设备被占用的操作都容易出现问题,这种情况下不需要继承的fd设置FD_CLOEXEC,会在调用exec函数的时候自动关闭

问题1:子进程可以继承父进程打开的文件描述符吗?

答:可以

  • 子进程获得父进程文件描述符表的副本
  • 相同的fd编号指向相同的文件表项

问题2:会因为父进程占用了而打不开吗?

答:不会

  • 每个进程有自己的文件描述符表
  • 多个进程可以同时打开同一个文件
  • 限制通常来自系统级(如打开文件总数限制)

问题3:子进程继承后,父进程关闭,子进程还在占用吗?

答:是的

  • 每个fd有独立的引用计数
  • 父进程关闭只减少自己那份的引用计数
  • 子进程的引用仍然存在

问题4:此时其它进程能占用吗?

答:能

  • 其他进程可以正常打开同一个文件
  • 文件是否真正被"占用"取决于文件锁

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 17:25:07

HoRain云--Python长连接实现:4种高效方案详解

&#x1f3ac; HoRain 云小助手&#xff1a;个人主页 ⛺️生活的理想&#xff0c;就是为了理想的生活! ⛳️ 推荐 前些天发现了一个超棒的服务器购买网站&#xff0c;性价比超高&#xff0c;大内存超划算&#xff01;忍不住分享一下给大家。点击跳转到网站。 目录 ⛳️ 推荐 …

作者头像 李华
网站建设 2026/6/10 16:21:11

校园跑腿|基于java+ vue校园跑腿系统(源码+数据库+文档)

校园跑腿 目录 基于springboot vue校园跑腿系统 一、前言 二、系统功能演示 三、技术选型 四、其他项目参考 五、代码参考 六、测试参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 基于springboot vue校园跑腿系统 一、前言 博主介绍&#xff1a;✌️大…

作者头像 李华
网站建设 2026/6/9 12:43:01

分布计算系统大题

本题要求对分布式系统中的共享资源问题与多副本数据问题进行综述。以下答案将根据您提供的资料进行组织和引用。五、综述题 1、哪些问题与共享资源相关&#xff0c;以及对应的解决办法&#xff1f; 在分布计算系统中&#xff0c;共享资源的使用是核心问题之一。由于资源的分散性…

作者头像 李华
网站建设 2026/6/9 10:37:12

Unity IL2CPP的GC原理

【USparkle专栏】如果你深怀绝技&#xff0c;爱“搞点研究”&#xff0c;乐于分享也博采众长&#xff0c;我们期待你的加入&#xff0c;让智慧的火花碰撞交织&#xff0c;让知识的传递生生不息&#xff01; 背景&#xff1a;前段时间在项目内做了关于Mono内存&#xff08;堆内存…

作者头像 李华
网站建设 2026/6/10 11:57:34

Word转PDF工具,学生作业一键转换上传更便捷换上传更便捷

&#x1f4cc;这些作业提交场景&#xff0c;真的太需要它&#xff01;&#xff01; 写好的Word作业排版混乱、老师要求PDF格式却没工具、在家写完作业要传给课代表却卡壳…是不是每次交作业都被格式问题绊住脚&#xff1f; 精心排版的语文作文&#xff0c;发给老师后字体错乱、…

作者头像 李华
网站建设 2026/6/10 16:32:14

Linux命令学习

Linux命令行 shell和终端 日常的基础应用都可以通过Linux系统的图形界面完成, 但界面做得再完善,还是有很多操作无法通过界面完成的,因为绝大部分程序本身根本就不提供界面, 或者界面只提供一小部分常用操作。但是几乎所有的程序都能通过命令行来调用运行, 并且通过命令…

作者头像 李华