Linux系统编程之exec函数簇的使用:剖析exec函数簇的实现原理、参数解释和用法技巧

简介: Linux系统编程之exec函数簇的使用:剖析exec函数簇的实现原理、参数解释和用法技巧

exec函数

在Linux中,并不存在exec()函数,exec指的是一组函数,一共有6个,分别是:

#include <unistd.h>
extern char **environ;
int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ..., char * const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);
int fexecve(int fd, char *const argv[], char *const envp[]);
//和execve执行相同的任务,文件描述符fd必须的只读并且调用者对此文件得有执行权限。

其中只有execve是真正意义上的系统调用,其它都是在此基础上经过包装的库函数。

exec函数族的作用是根据指定的文件名找到可执行文件,并用它来取代调用进程的内容,换句话说,就是在调用进程内部执行一个可执行文件。

对于exec系列函数一个进程一旦调用exec类函数,它本身就“死亡”了,系统把代码段替换成新的程序的代码,废弃原有的数据段和堆栈段,并为新程序分配新的数据段与堆栈段,唯一留下的,就是进程号,也就是说,对系统而言,还是同一个进程,不过已经是另一个程序了。不过exec类函数中有的还允许继承环境变量之类的信息,这个通过exec系列函数中的一部分函数的参数可以得到。

这里的可执行文件既可以是二进制文件,也可以是任何Linux下可执行的脚本文件。

这6个函数都是以exec开头(表示属于exec函数组),后缀l、v、p、e指定函数将具有某种操作能力:

后缀 操作能力
l list(列举参数) 希望接收以逗号分隔的参数列表,列表以NULL指针作为结束标志 列举参数 :const char *arg =“ls”,“-l”,NULL 采用了罗列(list)的方式,把参数一个一个列出来,然后以一个NULL表示结束。
v vector(参数向量表/字符串数组) 希望接收到一个以NULL结尾的字符串数组的指针 char *args[]={“ls”,“-l”,NULL}; char *const argv[]=args 以"char *argv[]"(vector)形式传递命令行参数
p 是一个以NULL结尾的字符串数组指针,函数可以DOS的环境变量PATH的目录里查找子程序文件
e 函数传递指定参数envp,允许改变子进程的环境, 无后缀e时,子进程使用当前程序的环境。

exec函数簇参数


path:必须是一个完整的路径,如"/bin/ls"//调用/bin目录下的ls命令;
file:可以仅仅只是一个文件名,如"ls",这两个函数可以自动到环境变量PATH指定的目录里去查找。
envp[]:指定当前进程所使用的环境变量


exec函数簇返回值

与一般情况不同,exec函数族的函数执行成功后不会返回,因为调用进程的实体,包括代码段,数据段和堆栈等都已经被新的内容取代,只有进程ID等一些表面上的信息仍保持原样。
调用失败时,会设置errno并返回-1,然后从原程序的调用点接着往下执行。
exec函数容易失败,最常见的错误:
找不到文件或路径,此时errno被设置为ENOENT
数组argv和envp忘记用NULL结束,此时errno被设置为EFAULT
没有对要执行文件的运行权限, 此时errno被设置为EACCES


exec函数簇示例

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/wait.h>
int main(void)
{
    int childpid;  
    int i;  
      
    if (fork() == 0){  
        //child process 
        char aszInputCmd[128] = {0};
        printf("Enter the calling method\n");
        int ret=read(fileno(stdin),aszInputCmd,sizeof(aszInputCmd));
        if(ret>0)
        {
          printf("execute function immediately from %s",aszInputCmd);
          if (0 == strncmp("quit", aszInputCmd, 4))
          {
            exit(0);
          }
          else if (0 == strncmp("execv", aszInputCmd, 5) )
          {
            char * execv_str[] = {"echo", "executed by execv",NULL}; 
            if (execv("/bin/echo",execv_str) <0 ){  
                    perror("error on exec");  
                    exit(0);  
                }  
            }
          }
          else if (0 == strncmp("execve", aszInputCmd, 6))
          {
            char * execve_str[] = {"env",NULL};  
            char * env[] = {"PATH=/tmp", "USER=lei", "STATUS=testing", NULL};  
            if (execve("/bin/env",execve_str,env) <0 ){  
                perror("error on exec");  
                exit(0);  
            }
          }
          else if (0 == strncmp("execvp", aszInputCmd, 6))
          {
            char * execvp_str[] = {"echo", "executed by execvp",">>", "~/abc.txt",NULL};  
            if (execvp("echo",execvp_str) <0 ){  
                perror("error on exec");  
                exit(0);  
            } 
          }
          else if (0 == strncmp("execl", aszInputCmd, 5) )
          {
            if (execl("/usr/bin/echo","echo","executed by execl" ,NULL) <0 ){  
                perror("error on exec");  
                exit(0);  
            } 
          }
          else if (0 == strncmp("execlp", aszInputCmd, 6) )
          {
            if (execlp("echo","echo","executed by execlp" ,NULL) <0 ){  
                perror("error on exec");  
                exit(0);  
            }  
          }
          else if (0 == strncmp("execle", aszInputCmd, 6) )
          {
            char * env[] = {"PATH=/home/gateman", "USER=lei", "STATUS=testing", NULL};  
            if (execle("/bin/env","env",NULL,env) <0){  
                perror("error on exec");  
                exit(0);  
            }   
          }
        }
     //parent process  
     wait(&childpid);  
     printf("execv test done\n\n");  
 }


目录
相关文章
|
6天前
|
Linux 应用服务中间件 Shell
linux系统服务二!
本文详细介绍了Linux系统的启动流程,包括CentOS 7的具体启动步骤,从BIOS自检到加载内核、启动systemd程序等。同时,文章还对比了CentOS 6和CentOS 7的启动流程,分析了启动过程中的耗时情况。接着,文章讲解了Linux的运行级别及其管理命令,systemd的基本概念、优势及常用命令,并提供了自定义systemd启动文件的示例。最后,文章介绍了单用户模式和救援模式的使用方法,包括如何找回忘记的密码和修复启动故障。
23 5
linux系统服务二!
|
6天前
|
Linux 应用服务中间件 Shell
linux系统服务!!!
本文详细介绍了Linux系统(以CentOS7为例)的启动流程,包括BIOS自检、读取MBR信息、加载Grub菜单、加载内核及驱动程序、启动systemd程序加载必要文件等五个主要步骤。同时,文章还对比了CentOS6和CentOS7的启动流程图,并分析了启动流程的耗时。此外,文中还讲解了Linux的运行级别、systemd的基本概念及其优势,以及如何使用systemd管理服务。最后,文章提供了单用户模式和救援模式的实战案例,帮助读者理解如何在系统启动出现问题时进行修复。
24 3
linux系统服务!!!
|
6天前
|
网络协议 Linux
linux系统重要文件目录
本文介绍了Linux系统中的重要目录及其历史背景,包括根目录、/usr、/etc、/var/log和/proc等目录的结构和功能。其中,/etc目录下包含了许多关键配置文件,如网卡配置、DNS解析、主机名设置等。文章还详细解释了各目录和文件的作用,帮助读者更好地理解和管理Linux系统。
23 2
|
5月前
|
消息中间件 存储 缓存
【嵌入式软件工程师面经】Linux系统编程(线程进程)
【嵌入式软件工程师面经】Linux系统编程(线程进程)
124 1
|
6月前
|
Linux 调度 数据库
Linux下的系统编程——线程同步(十三)
Linux下的系统编程——线程同步(十三)
111 0
Linux下的系统编程——线程同步(十三)
|
存储 Linux 调度
Linux系统编程 多线程基础
Linux系统编程 多线程基础
61 1
|
6月前
|
存储 安全 数据管理
Linux系统编程教程之Linux线程函数的使用:讲解Linux线程函数
Linux系统编程教程之Linux线程函数的使用:讲解Linux线程函数
62 1
|
6月前
|
NoSQL Unix Linux
Linux下的系统编程——守护进程、线程(十二)
Linux下的系统编程——守护进程、线程(十二)
70 0
Linux下的系统编程——守护进程、线程(十二)
|
6月前
|
安全 Linux C语言
Linux系统编程(线程同步 互斥锁)
Linux系统编程(线程同步 互斥锁)
67 0
|
6月前
|
设计模式 安全 Java
《Linux从练气到飞升》No.31 多线程编程实践与线程安全技术
《Linux从练气到飞升》No.31 多线程编程实践与线程安全技术
58 0