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 权限。
    • 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函数的更多相关文章

  1. 二十、Linux 进程与信号---非局部跳转

    20.1 setjmp 和 longjmp 函数 20.1.1 函数介绍 #include <setjmp.h> int setjmp(jmp_buf env); 函数功能:设置非局部跳转 ...

  2. 二十六、Linux 进程与信号---system 函数 和进程状态切换

    26.1 system 函数 26.1.1 函数说明 system(执行shell 命令)相关函数 fork,execve,waitpid,popen #include <stdlib.h> ...

  3. 三十、Linux 进程与信号——信号的概念及 signal 函数

    30.1 信号的基本概念 信号(signal)机制是Linux 系统中最为古老的进程之间的通信机制,解决进程在正常运行过程中被中断的问题,导致进程的处理流程会发生变化 信号是软件中断 信号是异步事件 ...

  4. 二十三、Linux 进程与信号---进程链和进程扇、守护进程和孤儿进程以及僵尸进程

    23.1 进程链和进程扇 23.1.1 概念 进程链:一个父进程构建出一个子进程,子进程再构建出子子进程,子子进程构建出子子子进程.... 这种就为进程链 进程扇:一个父进程构建出多个子进程,子进程都 ...

  5. 二十四、Linux 进程与信号---wait 函数

    24.1 wait 函数说明 24.1.1 waitpid---等待子进程中断或结束 waitpid(等待子进程中断或结束) 相关函数 wait,fork #include <sys/types ...

  6. 二十一、Linux 进程与信号---进程查看和进程状态、进程调度和进程状态变化、进程标识

    21.1 进程查看和进程状态 21.1.1 ps 指令 ps 指令通常可以查看到进程的 ID.进程的用户 ID.进程状态和进程的 Command ps:查看当前用户启动的进程 ps -ef:详细查看后 ...

  7. 二十一、Linux 进程与信号---进程资源限制

    21.1 进程资源限制 在操作系统中,我们能够通过函数getrlimit().setrlimit()分别获得.设置每个进程能够创建的各种系统资源的限制使用量. 21.1.1 函数 #include & ...

  8. 二十七、Linux 进程与信号---进程组和组长进程

    27.1 进程组 27.1.1 进程组介绍 进程组为一个或多个进程的集合 进程组可以接受同一终端的各种信号,同一个信号发送进程组等于发送给组中的所有进程 每个进程组有唯一的进程组 ID 进程组的消亡要 ...

  9. Linux学习之CentOS(二十六)--Linux磁盘管理:LVM逻辑卷的创建及使用

    在上一篇随笔里面 Linux学习之CentOS(二十五)--Linux磁盘管理:LVM逻辑卷基本概念及LVM的工作原理,详细的讲解了Linux的动态磁盘管理LVM逻辑卷的基本概念以及LVM的工作原理, ...

随机推荐

  1. Bomb HDU - 5934 (Tarjan)

    #include<map> #include<set> #include<ctime> #include<cmath> #include<stac ...

  2. 前端基础-- CSS

    CSS知识 CSS(Cascading Style Sheet,层叠样式表)定义如何显示HTML元素. 当浏览器读到一个样式表,它就会按照这个样式表来对文档进行格式化(渲染).Css之车更丰富的文档外 ...

  3. 使用Disruptor实现生产者和消费者模型

    生产者 package cn.lonecloud.procum.disruptor; import cn.lonecloud.procum.Data; import com.lmax.disrupto ...

  4. redux源码解析-redux的架构

    redux很小的一个框架,是从flux演变过来的,尽管只有775行,但是它的功能很重要.react要应用于生成环境必须要用flux或者redux,redux是flux的进化产物,优于flux. 而且r ...

  5. Codeforces Round #512 D - Vasya and Triangle

    D - Vasya and Triangle #include<bits/stdc++.h> using namespace std; #define LL long long LL gc ...

  6. ftp sun jdk自带

    package com.italktv.colnv.stat.util; import java.io.File; import java.io.FileInputStream; import jav ...

  7. ArcGIS for qml -设置地图和视域中心

    源码:https://github.com/sueRimn/ArcGIS-for-qml-demos 作者: 狐狸家的鱼 Github: 八至 版权声明:如需转载请获取授权和联系作者 ArcGIS R ...

  8. 前端JS Excel解析导入

    本文转载自:https://www.cnblogs.com/yinqingvip/p/6743213.html 需要用到js-xlsx:下载地址:js-xlsx <!DOCTYPE html&g ...

  9. typedef typename的用法

    我自己最后在这篇文章里理解:[C++]typedef typename什么意思? typedef typename A::a_type b_type; 其中typename是告诉编译器A::a_typ ...

  10. WORD2010如何把全角字母和数字批量转换成半角

    个人觉得全角字符看起来相当别扭,如果文档中存在大量全角形式的字母和数字,要如何把它们全部转化成半角的呢?   全角和半角   全角是指一个字符占用两个标准字符位置的状态.汉字字符和规定了全角的英文字符 ...