news 2026/4/16 3:45:36

学习笔记——进程控制函数

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
学习笔记——进程控制函数

进程控制函数整理笔记

一、 进程回收函数

wait() - 阻塞回收

#include <sys/wait.h> pid_t wait(int *status);

功能:阻塞等待任意子进程退出并回收状态(只能父进程回收子进程)

参数

  • status:进程退出时的状态

    • 不关心退出状态:NULL

    • 需要回收状态:传变量地址

返回值

  • 成功:回收的子进程PID

  • 失败:-1

状态宏使用

int status; pid_t pid = wait(&status); if (WIFEXITED(status)) { // 是否正常结束 int exit_code = WEXITSTATUS(status); // 获取退出码 printf("子进程正常退出,退出码: %d\n", exit_code); } else if (WIFSIGNALED(status)) { // 是否被信号终止 int sig_num = WTERMSIG(status); // 获取信号编号 printf("子进程被信号终止,信号: %d\n", sig_num); }

waitpid() - 精确控制回收

pid_t waitpid(pid_t pid, int *status, int options);

参数

  1. pid

    • >0:指定回收特定PID的子进程

    • -1:回收任意子进程(等价于wait()

    • 0:回收同进程组的子进程

    • <-1:回收指定进程组ID的绝对值

  2. status:同wait()

  3. options

    • 0:阻塞模式(默认)

    • WNOHANG:非阻塞模式

返回值

  • 成功:回收的子进程PID

  • 0WNOHANG模式下,子进程未退出

  • -1:错误

二、 非阻塞回收示例

方式1:循环检查

// 非阻塞回收所有子进程 int status; pid_t pid; while (1) { pid = waitpid(-1, &status, WNOHANG); if (pid > 0) { // 成功回收一个子进程 printf("回收子进程 %d\n", pid); if (WIFEXITED(status)) { printf("退出码: %d\n", WEXITSTATUS(status)); } } else if (pid == 0) { // 没有子进程退出,可以做其他事情 printf("没有子进程退出,继续工作...\n"); sleep(1); } else if (pid == -1) { // 错误或没有子进程了 printf("所有子进程已回收完毕\n"); break; } }

方式2:信号驱动(SIGCHLD)

#include <signal.h> void sigchld_handler(int sig) { int status; pid_t pid; while ((pid = waitpid(-1, &status, WNOHANG)) > 0) { printf("子进程 %d 已结束\n", pid); } } int main() { signal(SIGCHLD, sigchld_handler); // ... 创建子进程 }

三、 exec函数族

内存变化

执行exec前:原进程有自己的代码段、数据段、堆栈
执行exec后:原进程的代码段被新程序替换,PID不变,其他资源可能变化

函数对比表

函数路径指定参数传递PATH查找
execl()完整路径参数列表
execlp()文件名参数列表
execv()完整路径参数数组
execvp()文件名参数数组

函数详解

execl() - 参数列表形式
int execl(const char *path, const char *arg, ..., NULL);
  • path:完整路径(如"/usr/bin/ls"

  • arg:参数列表,以NULL结束

execl("/usr/bin/ls", "ls", "-l", "/home", NULL);
execlp() - 自动PATH查找
int execlp(const char *file, const char *arg, ..., NULL);
  • file:程序名,自动在PATH中查找

execlp("ls", "ls", "-l", "/home", NULL);
execv() - 参数数组形式
int execv(const char *path, char *const argv[]);
  • argv:参数数组,最后一项必须是NULL

char *args[] = {"ls", "-l", "/home", NULL}; execv("/usr/bin/ls", args);
execvp() - 自动PATH查找+数组参数
int execvp(const char *file, char *const argv[]);
char *args[] = {"ls", "-l", "/home", NULL}; execvp("ls", args);

调用自己的程序

// 假设当前目录有程序 myapp execl("./myapp", "myapp", "arg1", "arg2", NULL); // 必须带路径 execlp("./myapp", "myapp", "arg1", "arg2", NULL); // 带路径也可 execv("./myapp", args); // 必须带路径 execvp("./myapp", args); // 带路径也可

四、 fork + exec 配合使用

#include <stdio.h> #include <unistd.h> #include <sys/wait.h> int main() { pid_t pid = fork(); if (pid == 0) { // 子进程执行新程序 execlp("ls", "ls", "-l", NULL); // 如果exec失败才会执行到这里 perror("execlp failed"); exit(1); } else if (pid > 0) { // 父进程等待子进程 int status; waitpid(pid, &status, 0); if (WIFEXITED(status)) { printf("子进程退出码: %d\n", WEXITSTATUS(status)); } } return 0; }

五、 system()函数

#include <stdlib.h> int system(const char *command);

功能:执行shell命令(内部实现:fork() + exec()

限制

  • 不能执行需要修改父进程状态的命令(如cd,export等)

  • 适合执行信息输出、文件操作等命令

示例

system("ls -l"); // 列出文件 system("echo hello"); // 输出信息 system("cp file1 file2"); // 复制文件 // ❌ 这些无效(不会影响父进程): system("cd /tmp"); // 只在子shell中切换 system("export VAR=1"); // 只在子shell中设置

六、 工作目录函数

getcwd() - 获取当前工作目录

#include <unistd.h> char *getcwd(char *buf, size_t size);

参数

  • buf:存储路径的字符数组

  • size:数组最大长度

返回值

  • 成功:指向buf的指针

  • 失败:NULL

示例

char cwd[1024]; if (getcwd(cwd, sizeof(cwd)) != NULL) { printf("当前目录: %s\n", cwd); } else { perror("getcwd失败"); }

chdir() - 改变工作目录

int chdir(const char *path);

参数path- 要切换的路径

返回值

  • 成功:0

  • 失败:-1

示例

if (chdir("/tmp") != 0) { perror("chdir失败"); } else { printf("成功切换到 /tmp\n"); }

七、 关键点总结

  1. wait() vs waitpid()

    • wait():简单但只能阻塞回收

    • waitpid():功能更强,支持非阻塞和指定进程

  2. exec函数选择

    • 知道完整路径:用execl()execv()

    • 想用PATH查找:用execlp()execvp()

    • 参数动态:用execv()execvp()

    • 参数固定:用execl()execlp()

  3. 非阻塞回收

    • WNOHANG选项

    • 返回值需要特殊处理

    • 通常放在循环中

  4. cd命令特殊性

    • 必须在当前进程执行chdir()

    • 不能用fork()+exec()实现

    • 不能用system()实现

  5. 僵尸进程避免

    • 及时用wait()waitpid()回收

    • 对于后台进程,可以用信号处理SIGCHLD

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

微信小程序3D开发终极指南:threejs-miniprogram完整教程

微信小程序3D开发终极指南&#xff1a;threejs-miniprogram完整教程 【免费下载链接】threejs-miniprogram WeChat MiniProgram adapted version of Three.js 项目地址: https://gitcode.com/gh_mirrors/th/threejs-miniprogram 想要在微信小程序中实现惊艳的3D效果&…

作者头像 李华
网站建设 2026/4/14 14:27:16

Wan2.2-T2V-A14B视频生成模型实战:如何用140亿参数打造高保真T2V内容

Wan2.2-T2V-A14B视频生成模型实战&#xff1a;如何用140亿参数打造高保真T2V内容 在短视频日活破十亿、内容创作需求呈指数级增长的今天&#xff0c;一个广告片从创意到成片动辄数周&#xff0c;影视预演仍依赖手绘分镜和3D动画师逐帧调整——这种“人力密集型”生产模式正面临…

作者头像 李华
网站建设 2026/4/2 2:22:37

AutoGPT实战应用:让AI自主完成复杂任务的全流程解析

AutoGPT实战应用&#xff1a;让AI自主完成复杂任务的全流程解析 在信息爆炸的时代&#xff0c;我们每天面对的任务越来越复杂——从撰写一份详尽的市场分析报告&#xff0c;到为孩子制定个性化的学习计划&#xff0c;再到快速生成可运行的代码原型。传统的AI助手虽然能回答问题…

作者头像 李华
网站建设 2026/4/8 10:38:29

简单理解:SimpleFOC与正经 FOC(工业级)

SimpleFOC 是FOC&#xff08;磁场定向控制&#xff09;的开源简化实现版本&#xff0c;并非独立于 FOC 的技术&#xff0c;而你说的 “正经 FOC” 通常指工业级 / 商业级的标准 FOC 实现&#xff0c;二者核心都是 FOC 的磁场定向控制逻辑&#xff0c;但在实现复杂度、功能特性、…

作者头像 李华
网站建设 2026/4/8 11:52:09

小米运动刷步数终极教程:2025免费自动同步微信支付宝

小米运动刷步数终极教程&#xff1a;2025免费自动同步微信支付宝 【免费下载链接】mimotion 小米运动刷步数&#xff08;微信支付宝&#xff09;支持邮箱登录 项目地址: https://gitcode.com/gh_mirrors/mimo/mimotion 还在为每天运动步数不够而烦恼吗&#xff1f;想轻松…

作者头像 李华
网站建设 2026/4/13 10:18:00

【自然语言处理】中文 n-gram 词模型

目录 一、题目描述 二、解决方案 三、开发流程概述 &#xff08;一&#xff09;全局配置模块&#xff08;基础环境设置&#xff09; 核心功能&#xff1a; &#xff08;二&#xff09;文本预处理模块&#xff08;数据清洗与标准化&#xff09; 核心功能&#xff1a;将原始…

作者头像 李华