二十五、Linux 进程与信号---exec函数
25.1 介绍
- 在用 fork 函数创建子进程后,子进程往往要调用一种 exec 函数以执行另一个程序
- 当进程调用一种 exec 函数时,该进程完全由新程序代换,替换原有进程的正文,而新程序则从其 main 函数开始执行。因为调用 exec 并不创建新进程,所以前后的进程 ID 并未改变。exec 只是用另一个新程序替换了当前进程的正文、数据、堆和栈段。
- exec 函数族也称为代码替换函数族
25.1.1 函数说明
#include <unistd.h>
int execl(const char * path,const char * arg,....);
int execv (const char * path, char * const argv[ ]);
int execle(const char * path,const char * arg,....,char *const envp[]);
int execve(const char * filename,char * const argv[ ],char * const envp[ ]);
int execvp(const char *file ,char * const argv []);
int execlp(const char * file,const char * arg,……);
- 函数功能:
- execl()用来执行参数 path 字符串所代表的文件路径,接下来的参数代表执行该文件时传递过去的argv(0)、argv[1]……,最后一个参数必须用空指针(NULL)作结束。
execv() 用来执行参数 path 字符串所代表的文件路径,与 execl() 不同的地方在于 execve() 只需两个参数,第二个参数利用数组指针来传递给执行文件。
- execle() 是用来执行参数 path 字符串所代表的文件路径,并为新程序复制最后一个参数所指示的环境变量。接下来的参数代表执行该文件时传递过去的argv(0)、argv[1]……,最后一个参数必须用空指针(NULL)作结束。
- execve()用来执行参数 filename 字符串所代表的文件路径,第二个参数系利用数组指针来传递给执行文件,最后一个参数则为传递给执行文件的新环境变量数组。
- execvp()会从PATH 环境变量所指的目录中查找符合参数file 的文件名,找到后便执行该文件,然后将第二个参数argv传给该欲执行的文件。
- execlp() 会从 PATH 环境变量所指的目录中查找符合参 file 的文件名,找到后便执行该文件,然后将第二个以后的参数当做该文件的 argv[0]、argv[1]……,最后一个参数必须用空指针 (NULL) 作结束。
- 返回值
- 出错返回 -1,错误码存放,成功则不返回
- 错误码
- EACCES
- 欲执行的文件不具有用户可执行的权限。
- 欲执行的文件所属的文件系统是以 noexec 方式挂上。
- 欲执行的文件或 script 翻译器非一般文件。
- EPERM
- 进程处于被追踪模式,执行者并不具有 root 权限,欲执行的文件具有 SUID 或 SGID 位。
- 欲执行的文件所属的文件系统是以 nosuid 方式挂上,欲执行的文件具有 SUID 或 SGID 位元,但执行者并不具有 root 权限。
- EACCES
- E2BIG 参数数组过大
- ENOEXEC 无法判断欲执行文件的执行文件格式,有可能是格式错误或无法在此平台执行。
- EFAULT 参数 filename 所指的字符串地址超出可存取空间范围。
- ENAMETOOLONG 参数 filename 所指的字符串太长。
- ENOENT 参数 filename 字符串所指定的文件不存在。
- ENOMEM 核心内存不足
- ENOTDIR 参数 filename 字符串所包含的目录路径并非有效目录
- EACCES 参数 filename 字符串所包含的目录路径无法存取,权限不足
- ELOOP 过多的符号连接
- ETXTBUSY 欲执行的文件已被其他进程打开而且正把数据写入该文件中
- EIO I/O 存取错误
- ENFILE 已达到系统所允许的打开文件总数。
- EMFILE 已达到系统所允许单一进程所能打开的文件总数。
- EINVAL 欲执行文件的ELF执行格式不只一个PT_INTERP节区
- EISDIR ELF翻译器为一目录
- ELIBBAD ELF翻译器有问题
- 注意点
- execve 函数为系统调用,其余为库函数。执行 execve 函数后面的代码不执行
- execlp 和 execvp 函数中的 path,相对和绝对路径均可使用,其他四个函数中的 path 只能使用绝对路径。相对路径一定要在进程环境表对应的 PATH 中。
- argv 参数为新程序执行 main 函数中传递的 argv 参数,最后一个元素为 NULL
- envp 为进程环境表
- 六个函数都使以 "exec"四个字母开头的,后面的字母表示了其用法上的区别:
- 带有字母 " l " 的函数,表明后面的参数列表是要传递给程序的参数列表,参数列表的第一个参数必须是 要执行的程序,最后一个参数必须是 NULL
- 带有字母 “ p ”的函数,第一个参数可用是相对路径或程序名,如果无法立即找到要执行的程序,那么就在环境变量 PATH 指定的路径中搜索。其他函数的第一个参数必须是绝对路径
- 带有字母 " v "的函数,表明程序的参数列表通过一个字符串数组来传递。这个数组和最后传递给程序的 main 函数的字符串数据 argv 完全一样。第一个参数必须是程序名,最后一个参数也必须是 NULL
- 带有字母 " e " 的函数,用户可用自己设置程序接收一个设置环境变量的数组
六个函数之间的关系如下:

25.2 例子
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h> char *cmd1 = "cat"; //相对路径
char *cmd2 = "/bin/cat";//绝对路径
char *argv1 = "/etc/passwd";
char *argv2 = "/etc/group"; int main(void)
{
pid_t pid;
if((pid = fork()) < ){
perror("fork error");
exit();
} else if(pid == ) {
//子进程调用 exec 函数执行新的程序
//execl 不带 p 需要用绝对路径的
if(execl(cmd2, cmd1, argv1, argv2, NULL) < ) {
perror("execl error");
exit();
} else {
printf("execl %s success\n", cmd1);
}
} wait(NULL);
printf("=================================\n"); if((pid = fork()) < ){
perror("fork error");
exit();
} else if(pid == ) { char *argv[] = {cmd1, argv1, argv2, NULL};
if(execvp(cmd1, argv) < ) {
perror("execl error");
exit();
} else {
printf("execvp %s success\n", cmd1);
}
} wait(NULL);
printf("=================================\n");
return ;
}
二十五、Linux 进程与信号---exec函数的更多相关文章
- 二十、Linux 进程与信号---非局部跳转
20.1 setjmp 和 longjmp 函数 20.1.1 函数介绍 #include <setjmp.h> int setjmp(jmp_buf env); 函数功能:设置非局部跳转 ...
- 二十六、Linux 进程与信号---system 函数 和进程状态切换
26.1 system 函数 26.1.1 函数说明 system(执行shell 命令)相关函数 fork,execve,waitpid,popen #include <stdlib.h> ...
- 三十、Linux 进程与信号——信号的概念及 signal 函数
30.1 信号的基本概念 信号(signal)机制是Linux 系统中最为古老的进程之间的通信机制,解决进程在正常运行过程中被中断的问题,导致进程的处理流程会发生变化 信号是软件中断 信号是异步事件 ...
- 二十三、Linux 进程与信号---进程链和进程扇、守护进程和孤儿进程以及僵尸进程
23.1 进程链和进程扇 23.1.1 概念 进程链:一个父进程构建出一个子进程,子进程再构建出子子进程,子子进程构建出子子子进程.... 这种就为进程链 进程扇:一个父进程构建出多个子进程,子进程都 ...
- 二十四、Linux 进程与信号---wait 函数
24.1 wait 函数说明 24.1.1 waitpid---等待子进程中断或结束 waitpid(等待子进程中断或结束) 相关函数 wait,fork #include <sys/types ...
- 二十一、Linux 进程与信号---进程查看和进程状态、进程调度和进程状态变化、进程标识
21.1 进程查看和进程状态 21.1.1 ps 指令 ps 指令通常可以查看到进程的 ID.进程的用户 ID.进程状态和进程的 Command ps:查看当前用户启动的进程 ps -ef:详细查看后 ...
- 二十一、Linux 进程与信号---进程资源限制
21.1 进程资源限制 在操作系统中,我们能够通过函数getrlimit().setrlimit()分别获得.设置每个进程能够创建的各种系统资源的限制使用量. 21.1.1 函数 #include & ...
- 二十七、Linux 进程与信号---进程组和组长进程
27.1 进程组 27.1.1 进程组介绍 进程组为一个或多个进程的集合 进程组可以接受同一终端的各种信号,同一个信号发送进程组等于发送给组中的所有进程 每个进程组有唯一的进程组 ID 进程组的消亡要 ...
- Linux学习之CentOS(二十六)--Linux磁盘管理:LVM逻辑卷的创建及使用
在上一篇随笔里面 Linux学习之CentOS(二十五)--Linux磁盘管理:LVM逻辑卷基本概念及LVM的工作原理,详细的讲解了Linux的动态磁盘管理LVM逻辑卷的基本概念以及LVM的工作原理, ...
随机推荐
- poj2373 Dividing the Path (单调队列+dp)
题意:给一个长度为L的线段,把它分成一些份,其中每份的长度∈[2A,2B]且为偶数,而且不能在某一些区间内部切开,求最小要分成几份 设f[i]为在i处切一刀,前面的满足要求的最小份数,则f[L]为答案 ...
- Luogu P1648 看守
Luogu P1648 看守 题意简述 有n个d维的点,输出这些点两两之间曼哈顿距离中的最大值 数据范围 n<=1e6,d<=4 思路 暴力?时间复杂度O(\(n^2d\)) 考虑这样的一 ...
- webpack入门(五)webpack CLI
webpack的CLI安装和命令 Installation $ npm install webpack -g The webpack command is now available globally ...
- nodejs的某些api~(五) HTTP模块
HTTP的模块是nodejs最重要的模块(应该是),最近在看HTTP权威指南,重新过了一遍http协议和web客户端.再来看这个http. HTTP构建于TCP之上,属于应用层协议,继承自tcp服务器 ...
- HDU--4486 Task(贪心)
题目链接 4486 Task 按照时间从大到小排序 然后枚举所有的y值 用一个数组存储 符合要求就算上 #include<bits/stdc++.h> using namespace s ...
- Spring 官方教程:使用 Restdocs 创建 API 文档
https://mp.weixin.qq.com/s?__biz=MzU0MDEwMjgwNA==&mid=2247483998&idx=1&sn=6ae5fa795d36b1 ...
- [HEOI2014]平衡
[HEOI2014]平衡 转化为求选择k个数,和为(n+1)*k的方案数 保证,每个数[1,2*n+1]且最多选择一次. 限制k个很小,所以用整数划分的第二种方法 f[i][j],用了i个,和为j 整 ...
- [luogu3939][数颜色]
题目链接 思路 对于每一种颜色都建立一个动态开点线段树.然后每次查询的时候就去这个颜色的线段树上查询就行了.修改之后不要忘记交换颜色. 这个题目数据有点强.抄了个比较快的读入优化才卡过去. 代码 /* ...
- linux free命令
Linux上的free命令详解 free命令的所有输出值都是从/proc/meminfo中读出的 total used free shared buffers cached Mem: -/+ buff ...
- struts2 二: 参数封装
封装请求正文到对象中(非常重要) 1.静态参数封装 在struts.xml配置文件中,给动作类注入值.调用的是setter方法. struts.xml的编写: <action name=&quo ...