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

exec进程替换函数族

excel是进程替换函数,在进行excel后它继承了原进程的一些资源,又丢弃了原进程的一些资源。

新进程与原始进程的相同点:

  • 进程 ID 在进程替换后保持不变。
  • 原进程中打开的文件描述符在新进程中默认保持打开状态,这意味着新进程可以继续使用原进程打开的文件、管道、套接字等资源。
  • 进程所属的进程组和会话在 execl 后不会改变。
  • 原进程设置的信号掩码会被新进程继承。信号掩码定义了哪些信号当前被阻塞而不会被进程立即处理。例如,如果原进程阻塞了 SIGINT 信号,调用 execl 后新进程同样会阻塞 SIGINT 信号,直到新进程显式地修改信号掩码。

新进程与原始进程的不同点:

  • execl 用新程序的代码段、数据段、堆栈段完全替换原进程相应的部分。原进程的代码不再执行,新进程从新程序的入口点开始执行。
  • 虽然信号掩码被继承,但原进程设置的信号处理函数在新进程中不再有效。
  • 如果在调用 execl 时没有显式设置环境变量(execle 和 execve 函数可设置),新进程会继承原进程的环境变量。然而,如果原进程在 execl 之前修改了环境变量,这些修改对新进程的影响取决于新程序对环境变量的依赖和使用方式。而且,如果使用 execle 或 execve 并设置了自定义的环境变量,那么新进程将使用这些特定的环境变量,与原进程的环境变量不同。(这一点在后续代码中实操)。如果大家不是很了解环境变量,可以看这篇文章:通俗易懂linux环境变量-CSDN博客

int execl(const char *path, const char *arg, ...)

  • 功能execl 函数用于执行一个指定路径的可执行文件,并传递可变参数列表给该可执行文件的 main 函数。
  • 参数
    • path:指定要执行的可执行文件的完整路径。例如,/bin/ls 表示要执行系统中的 ls 命令程序。
    • arg:可变参数列表,以 NULL 结尾,注意这里最后一个参数必须是NULL。这些参数会被传递给新程序的 main 函数。第一个参数通常是可执行文件的名称。

        示例代码如下,输出的话,大家自行测试即可:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork");return 1;} else if (pid == 0) {// 子进程if (execl("/bin/ls", "ls", "-l", "/tmp", NULL) == -1) {perror("execl");exit(1);}} else {// 父进程wait(NULL);printf("Parent process continues...\n");}return 0;
}

int execlp(const char *file, const char *arg, ...)

  • 功能:与 execl 类似,但 execlp 会在环境变量 PATH 中搜索要执行的可执行文件,而不是需要完整路径。
  • 参数
    • file:要执行的可执行文件的名称,系统会在 PATH 变量指定的目录中查找该文件。
    • arg:可变参数列表,同样以 NULL 结尾,传递给新程序的 main 函数。

        示例代码如下,输出的话,大家自行测试即可:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork");return 1;} else if (pid == 0) {// 子进程if (execlp("ls", "ls", "-l", "/tmp", NULL) == -1) {perror("execlp");exit(1);}} else {// 父进程wait(NULL);printf("Parent process continues...\n");}return 0;
}

int execle(const char *path, const char *arg, ..., char *const envp[])

  • 功能execle 函数用于执行指定路径的可执行文件,同时可以为新程序设置自定义的环境变量。
  • 参数
    • path:可执行文件的完整路径。
    • arg:可变参数列表,传递给新程序的 main 函数,以 NULL 结尾。
    • envp:一个以 NULL 结尾的字符串数组,该数组必须以NULL结尾,用于设置新程序的环境变量。每个字符串的格式为 "变量名=值"

        示例代码如下,输出的话,大家自行测试即可:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork");return 1;} else if (pid == 0) {// 子进程char *envp[] = {"LD_LIBRARY_PATH=/my/lib/path", NULL};if (execle("/my/program", "program", NULL, envp) == -1) {perror("execle");exit(1);}} else {// 父进程wait(NULL);printf("Parent process continues...\n");}return 0;
}

        还记得前面所讲的吗:“如果使用 execle 或 execve 并设置了自定义的环境变量,那么新进程将使用这些特定的环境变量,与原进程的环境变量不同”。上述代码想必已经给大家做了解释。

int execv(const char *path, char *const argv[])

  • 功能execv 函数执行指定路径的可执行文件,并通过一个字符串数组传递参数给新程序的 main 函数。
  • 参数
    • path:可执行文件的完整路径。
    • argv:一个以 NULL 结尾的字符串数组,包含传递给新程序 main 函数的参数。

        示例代码如下,输出的话,大家自行测试即可:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork");return 1;} else if (pid == 0) {// 子进程char *argv[] = {"ls", "-l", "/tmp", NULL};if (execv("/bin/ls", argv) == -1) {perror("execv");exit(1);}} else {// 父进程wait(NULL);printf("Parent process continues...\n");}return 0;
}

int execvp(const char *file, char *const argv[])

  • 功能:与 execv 类似,但 execvp 会在环境变量 PATH 中搜索要执行的可执行文件。
  • 参数
    • file:要执行的可执行文件的名称。
    • argv:以 NULL 结尾的字符串数组,传递给新程序的 main 函数。

        示例代码如下,输出的话,大家自行测试即可:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork");return 1;} else if (pid == 0) {// 子进程char *argv[] = {"ls", "-l", "/tmp", NULL};if (execvp("ls", argv) == -1) {perror("execvp");exit(1);}} else {// 父进程wait(NULL);printf("Parent process continues...\n");}return 0;
}

int execve(const char *path, char *const argv[], char *const envp[])

  • 功能execve 是 exec 系列函数的基础,其他函数本质上是对它的封装。它执行指定路径的可执行文件,同时可以设置自定义的环境变量,并通过字符串数组传递参数。
  • 参数
    • path:可执行文件的完整路径。
    • argv:以 NULL 结尾的字符串数组,传递给新程序的 main 函数。
    • envp:以 NULL 结尾的字符串数组,用于设置新程序的环境变量。

        示例代码如下,输出的话,大家自行测试即可:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork");return 1;} else if (pid == 0) {// 子进程char *argv[] = {"ls", "-l", "/tmp", NULL};char *envp[] = {"LD_LIBRARY_PATH=/my/lib/path", NULL};if (execve("/bin/ls", argv, envp) == -1) {perror("execve");exit(1);}} else {// 父进程wait(NULL);printf("Parent process continues...\n");}return 0;
}

http://www.lqws.cn/news/215821.html

相关文章:

  • 使用 Redisson 实现分布式锁—解决方案详解
  • CMake检测C编译器标志功能
  • gopool 源码分析
  • 跨平台资源下载工具:res-downloader 的使用体验
  • electron-vite串口通信
  • 第9篇:数据库中间件的容错机制与高可用架构设计
  • 阿里云 RDS mysql 5.7 怎么 添加白名单 并链接数据库
  • 34 C 语言字符串转数值函数详解:strtol、strtoll、strtoul、strtoull(含 errno 处理、ERANGE 错误)
  • 坚持每日Codeforces三题挑战:Day 4 - 题目详解(2025-06-07,难度:1000, 1100, 1400)
  • 6个月Python学习计划 Day 16 - 面向对象编程(OOP)基础
  • 利用qcustomplot绘制曲线图
  • 智慧零售管理中的客流统计与属性分析
  • 前端开发者常用网站
  • 基于51单片机的光强调节LED亮度
  • 关于脏读,幻读,可重复读的学习
  • python训练营打卡第47天
  • CAD2025安装教程与资源下载
  • 湖北理元理律师事务所:债务咨询中的心理支持技术应用
  • cv2.stereoRectify中R1, R2, P1, P2, Q中每一个分量的物理意义
  • SMC自修改代码一
  • MCP详解及协议的使用(python版本和Node版本)
  • FreeRTOS事件组-笔记
  • AI大神吴恩达-提示词课程笔记
  • 【Go语言基础【14】】defer与异常处理(panic、recover)
  • 深入剖析MySQL存储架构,索引结构,日志机制,事务提交流程
  • 【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器
  • KTO: Model Alignment as Prospect Theoretic Optimization
  • [总结篇]个人网站
  • XGBoost时间序列预测之-未来销量的预测
  • 【氧化镓】HTFB应力对β - Ga2O3 SBD的影响