当前位置: 首页 > news >正文

linux--------------进程控制(下)

一、进程等待

1.1 进程等待必要性

  • 子进程退出后,若父进程不管不顾,可能会产生 “僵尸进程”,进而造成内存泄漏。
  • 进程一旦变为僵尸状态,即使使用 kill -9 也无法将其杀死,因为无法杀死一个已死的进程。
  • 父进程需要了解子进程的任务完成情况,比如子进程运行结束后结果是否正确,是否正常退出。
  • 父进程通过进程等待的方式,回收子进程资源并获取其退出信息。

1.2 进程等待的方法

wait() 函数

wait() 函数用于阻塞等待子进程的结束,并回收其资源。以下是一个简单的示例代码:

#include <sys/wait.h>
#include <stdio.h>

int main() {
    pid_t child_pid = fork();
    if (child_pid == 0) {
        // 子进程执行任务
        exit(42);  // 子进程退出码42
    } else {
        int status;
        pid_t terminated_pid = wait(&status);  // 阻塞等待
        if (WIFEXITED(status)) {
            printf("子进程 %d 退出码: %d\n", terminated_pid, WEXITSTATUS(status));
        }
    }
    return 0;
}
waitpid() 函数

waitpid() 函数的原型为 pid_t waitpid(pid_t pid, int *status, int options);,它可以更灵活地等待指定子进程的结束。

1.3 获取子进程 status

wait 和 waitpid 都有一个 status 参数,这是一个输出型参数,由操作系统填充。若传递 NULL,表示不关心子进程的退出状态信息;否则,操作系统会根据该参数将子进程的退出信息反馈给父进程。

status 不能简单地当作整型来看待,可将其视为位图,具体细节可参考下面的图片(只研究 status 低 16 比特位):

状态位说明
低8位(0-7)子进程退出码 (正常终止时有效)
第8位(8-15)信号编号 (被信号终止时有效)
其他标志位通过宏检测状态类型(如WIFEXITED)

1.4 阻塞和非阻塞

核心结论
  • 本质区别

    • 阻塞(Blocking):调用者线程暂停执行,直到操作完成(如数据到达、资源就绪)。

    • 非阻塞(Non-blocking):调用立即返回,无论操作是否完成,需通过轮询或事件通知获取结果。

  • 选择依据

    • 阻塞:适合简单逻辑、单任务场景,代码直观但资源利用率低。

    • 非阻塞:适合高并发、实时响应需求,需配合多路复用(如epoll)或异步通知(如回调)

深度解析
运行机制对比
特性阻塞模式非阻塞模式
线程状态挂起(Sleeping)持续运行(Running)
CPU 占用低(等待时不消耗 CPU)高(需轮询检查状态)
响应延迟操作完成后立即响应需主动检测或等待通知
代码复杂度低(线性执行)高(需处理中间状态和错误码)
典型应用场景

以下是阻塞和非阻塞模式的典型应用代码示例:

阻塞模式示例

#include <stdio.h>
#include <fcntl.h>

int main() {
    int fd = open("file.txt", O_RDONLY);  
    char buf[1024];  
    read(fd, buf, sizeof(buf));  // 阻塞直到数据就绪  
    printf("Data: %s\n", buf);  
    return 0;
}

非阻塞模式示例

#include <stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>

#define BUFFER_SIZE 1024

int main() {
    int sockfd = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0);  
    struct sockaddr_in server_addr;
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(8080);
    server_addr.sin_addr.s_addr = INADDR_ANY;

    connect(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr));  

    char data[BUFFER_SIZE] = "Hello, Server!";
    int len = strlen(data);

    while (1) {  
        if (send(sockfd, data, len, MSG_DONTWAIT) == -1) {  
            if (errno == EAGAIN) {  
                usleep(1000);  // 数据未就绪,短暂等待后重试  
                continue;  
            }  
        }  
        break;  
    }  

    close(sockfd);
    return 0;
}

二、进程程序替换

2.1 替换原理

使用 fork 创建子进程后,子进程执行的是与父进程相同的程序(但可能执行不同的代码分支)。若子进程想执行一个全新的程序,可通过进程的程序替换来实现。当进程调用一种 exec 函数时,该进程的用户空间代码和数据会完全被新程序替换,并从新程序的启动例程开始执行。调用 exec 并不会创建新进程,因此调用前后该进程的 ID 不会改变。

2.2 替换函数

函数名参数传递方式PATH 搜索环境变量典型用途
execl参数列表(可变参数)继承当前环境已知绝对路径的固定参数调用
execv参数数组(char *[]继承当前环境动态构建参数的固定路径调用
execlp参数列表继承当前环境执行 PATH 中的命令(如 Shell)
execvp参数数组继承当前环境动态执行 PATH 中的命令
execle参数列表自定义环境变量需严格控制环境的场景
execvpe参数数组自定义环境变量动态参数 + 自定义环境

2.3 函数解释

这些函数如果调⽤成功则加载新的程序从启动代码开始执⾏,不再返回。
如果调⽤出错则返回-1
所以exec函数只有出错的返回值⽽没有成功的返回值

2.4 命名理解

  • l(list):表示参数采用列表形式。
  • v(vector):参数使用数组。
  • p(path):有 p 则自动搜索环境变量 PATH
  • e(env):表示自己维护环境变量。

以下是调用示例:

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

int main() {
    char *const argv[] = {"ps", "-ef", NULL};
    char *const envp[] = {"PATH=/bin:/usr/bin", "TERM=console", NULL};

    execl("/bin/ps", "ps", "-ef", NULL);
    // 带 p 的,可以使用环境变量 PATH,无需写全路径
    execlp("ps", "ps", "-ef", NULL);
    // 带 e 的,需要自己组装环境变量
    execle("ps", "ps", "-ef", NULL, envp);
    execv("/bin/ps", argv);
    // 带 p 的,可以使用环境变量 PATH,无需写全路径
    execvp("ps", argv);
    // 带 e 的,需要自己组装环境变量
    execve("/bin/ps", argv, envp);

    // 如果 exec 调用失败,会执行到这里
    perror("exec 调用失败");
    return 1;
}

相关文章:

  • WPF如何修改三方控件库的样式
  • AudioRecord 录制pcm转wav
  • 每日一题(小白)数组娱乐篇17
  • 滑动窗口7:30. 串联所有单词的子串
  • 分布式数据库LSM树
  • 多模态大语言模型arxiv论文略读(七)
  • Unity-Xlua热更和AssetBundle详解
  • 上下拉电阻详解
  • RAG 系统中的偏差是什么?
  • 自定义数据结构的QVariant序列化 ASSERT failure in QVariant::save: “invalid type to save“
  • BetaFlight参数配置解读
  • 软考高项-考前冲刺资料-M 类【项目管理类】【光头张老师出品】
  • C++:模拟实现string
  • spring之Bean的循环依赖问题、反射机制手写Spring框架、Spring IoC注解式开发
  • 测试用例设计
  • 对抗Prompt工程:构建AI安全护栏的攻防实践
  • 精密空调的介绍
  • 《解码 C/C++ 关键字:科技编程的核心指令集》
  • 机器学习 Day09 线性回归
  • 在SQLark 中快速生成测试数据
  • 阿里CEO:将以饱和式投入打法,聚焦几大核心战役
  • 第一集|《刑警的日子》很生活,《执法者们》有班味
  • 专访|高圆圆:像鸟儿一样,柔弱也自由
  • 技术派|伊朗展示新型弹道导弹,美“萨德”系统真的拦不住?
  • 上海充电桩调研:须全盘考量、分步实现车网互动规模化
  • 牛市早报|央行宣布降准降息,公募基金改革最新方案落地