exec系列函数(execl,execlp,execle,execv,execvp)使用

本节目标:

exec替换进程映像
exec关联函数组(execl、execlp、execle、execv、execvp)
一,exec替换进程映像
在进程的创建上Unix采用了一个独特的方法,它将进程创建与加载一个新进程映象分离。这样的好处是有更多的余地对两种操作进行管理。

当我们创建了一个进程之后,通常将子进程替换成新的进程映象,这可以用exec系列的函数来进行。当然,exec系列的函数也可以将当前进程替换掉。

例如:在shell命令行执行ps命令,实际上是shell进程调用fork复制一个新的子进程,在利用exec系统调用将新产生的子进程完全替换成ps进程。

二,exec系列函数(execl、execlp、execle、execv、execvp)
包含头文件<unistd.h>

功能:

    用exec函数可以把当前进程替换为一个新进程,且新进程与原进程有相同的PID。exec名下是由多个关联函数组成的一个完整系列,

头文件<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[]);

参数:

path参数表示你要启动程序的名称包括路径名

arg参数表示启动程序所带的参数,一般第一个参数为要执行命令名,不是带路径且arg必须以NULL结束

返回值:成功返回0,失败返回-1

注:上述exec系列函数底层都是通过execve系统调用实现:

       #include <unistd.h>

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

DESCRIPTION:
       execve() executes the program pointed to by filename.  filename must be
       either a binary executable, or a script starting with  a  line  of  the form 

以上exec系列函数区别:

1,带l 的exec函数:execl,execlp,execle,表示后边的参数以可变参数的形式给出且都以一个空指针结束。

示例:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(void)
{
    printf("entering main process---\n");
    execl("/bin/ls","ls","-l",NULL);
    printf("exiting main process ----\n");
    return 0;
}利用execl将当前进程main替换掉,所有最后那条打印语句不会输出


2,带 p 的exec函数:execlp,execvp,表示第一个参数path不用输入完整路径,只有给出命令名即可,它会在环境变量PATH当中查找命令
示例:
当不带p但没给出完整路径时:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(void)
{
    printf("entering main process---\n");
    if(execl("ls","ls","-l",NULL)<0)
      perror("excl error");
    return 0;
}

结果显示找不到,所有替换不成功,main进程继续执行

现在带p:

if(execlp("ls","ls","-l",NULL)<0)

替换成功
3,不带 l 的exec函数:execv,execvp表示命令所需的参数以char *arg[]形式给出且arg最后一个元素必须

是NULL

示例:

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

int main(void)
{
    printf("entering main process---\n");
    int ret;
    char *argv[] = {"ls","-l",NULL};
    ret = execvp("ls",argv);
    if(ret == -1)
        perror("execl error");
    printf("exiting main process ----\n");
    return 0;
}替换成功


4,带 e 的exec函数:execle表示,将环境变量传递给需要替换的进程

从上述的函数原型中我们发现:

extern char **environ;

此处的environ是一个指针数组,它当中的每一个指针指向的char为“XXX=XXX”

environ保存环境信息的数据可以env命令查看:它由shell进程传递给当前进程,再由当前进程传递给替换的新进程
示例:execle.c

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

int main(int argc, char *argv[])
{
    //char * const envp[] = {"AA=11", "BB=22", NULL};
    printf("Entering main ...\n");
    int ret;
    ret =execl("./hello", "hello", NULL);
    //execle("./hello", "hello", NULL, envp);
    if(ret == -1)
        perror("execl error");
    printf("Exiting main ...\n");
    return 0;
}

hello.c

#include <unistd.h>
#include <stdio.h>
extern char** environ;

int main(void)
{
    printf("hello pid=%d\n", getpid());
    int i;
    for (i=0; environ[i]!=NULL; ++i)
    {
        printf("%s\n", environ[i]);
    }
    return 0;
}

可知原进程确实将环境变量信息传递给了新进程


那么现在我们可以利用execle函数自己给的需要传递的环境变量信息:

 


示例程序:execle.c

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

int main(int argc, char *argv[])
{
    char * const envp[] = {"AA=11", "BB=22", NULL};
    printf("Entering main ...\n");
    int ret;
    //ret =execl("./hello", "hello", NULL);
    ret =execle("./hello", "hello", NULL, envp);
    if(ret == -1)
        perror("execl error");
    printf("Exiting main ...\n");
    return 0;
}

hello.c

#include <unistd.h>
#include <stdio.h>
extern char** environ;

int main(void)
{
    printf("hello pid=%d\n", getpid());
    int i;
    for (i=0; environ[i]!=NULL; ++i)
    {
        printf("%s\n", environ[i]);
    }
    return 0;
}确实将给定的环境变量传递过来了

http://bbs.csdn.net/topics/350057686

要怎样做才能正常执行(主要是execl函数运行php文件。),并输入exit退出?
execl会载入你调用的程序,覆盖原有代码段,相当于你本来的程序的代码段被替换成execl执行的了。
所以execl后面的都不会输出了。
正确的应该是fork一个子进程,在子进程中调用execl。

如果我把execl 放到用pthread_create创建的线程是不是也一样会覆盖原有的代码?
我测试过也是一样会退出。。
一定要用fork 吗?
execl是覆盖进程的代码段,所以如果你原来的程序还需要正常退出的话,就要fork一个子进程。
另外,如果需要execl调用的这个程序执行完成,就需要在父进程中等待,可以调用waitpid。
具体的你man一下看看。

如果用了fork 。执行execl 后会不会停止 子进程的呢?
fork以后,子进程拷贝父进程的几乎所有特性,包括pc,也就是说,在子进程中,还是从你fork的地方开始运行的。然后你execl,覆盖掉的是子进程的代码段,与父进程无关。

pthread_create创建的子线程与父线程共用同一代码段!只是寄存器和栈不同!
execl不是创建一个新进程,Linux只有fork能创建新进程
execl成功是不会返回的,会执行新的程序。
execl失败才会返回

http://bbs.chinaunix.net/thread-4168834-1-1.html

我感觉execl的第一个参数和第二个参数有重叠,帮我 ...
如题,我学习了一下execl函数的用法:
#include<unistd.h>
int main()
{
  execl("ls", "ls", "-al");
  return 0;
}
OK,什么也不打印,于是我改了一下:
#include<unistd.h>
int main()
{
  execl("/bin/ls", "ls", "-al", 0);
  return 0;
}
很奇怪啊,execl第一个参数必须是path,我如果把调用语句改成execl("/bin", "ls", "-al", 0);
发现也不行。那岂不是说这个path参数必须包含可执行程序的名字本身? 可是这样的话,execl的第二个参数就是文件名啊,第二个参数岂不是多余了?

非常疑惑,求解释!
path里面包含了file的名字,那还要第二个file参数干什么呢? 感觉多余?
第2个参数将传给进程,也就是argv[0],进程可以通过查看面子有多大,就能看人下菜碟-哦是随机应变了!。
果然是高手!我验证了你的说法,第二个参数是argv[0],可以是任何东西,不必是可执行文件名。如下图所示:


三,fcntl()函数中的FD_CLOEXEC标识在exec系列函数中的作用


#include <unistd.h> 
#include <fcntl.h>


int fcntl(int fd, int cmd, ... /* arg */ );


File descriptor flags 
      The following commands manipulate the  flags  associated  with  a  file 
      descriptor.   Currently, only one such flag is defined: FD_CLOEXEC, the 
      close-on-exec flag.  If the FD_CLOEXEC bit is 0,  the  file  descriptor 
      will remain open across an execve(2), otherwise it will be closed.


//如果FD_CLOEXEC标识位为0,则通过execve调用后fd依然是打开的,否则为关闭的


F_GETFD (void) 
             Read the file descriptor flags; arg is ignored.


F_SETFD (long) 
             Set the file descriptor flags to the value specified by arg.


如:fcntl(fd, F_SETFD, FD_CLOEXEC);

1关闭(标准输出关闭)ls -l无法将结果显示在标准输出
http://www.cnblogs.com/mickole/p/3187409.html
				
时间: 2024-08-27 02:12:14

exec系列函数(execl,execlp,execle,execv,execvp)使用的相关文章

5进程原语:execl(),execlp(),execle(),execv(),execvp(),execvp(),execve()

 1.exec族依赖的头文件 #include<unistd.h> extern char**environ; 2函数声明 //第一个参数:可执行程序所在的绝对位置 //第二个参数:可执行程序运行时所需的参数,这里是一个可变参数,每个参数之间用逗号分割 //参数结束时,最后一个参数是:NULL int execl(constchar *path, const char *arg, ...); //注意,这里函数名加了p,表示调用了系统的环境变量中配置的PATH,查看环境变量命令:Echo

一起talk C栗子吧(第一百三十五回:C语言实例--exec系列函数一)

各位看官们,大家好,上一回中咱们说的是"创建线程时的内存细节"的例子,这一回咱们说的例子是:exec系列函数.闲话休提,言归正转.让我们一起talk C栗子吧! 看官们,我们在前面介绍创建进程内存细节的章回中提到过exec系列函数,当时并没有做详细的介绍.今天我们将和大家一起看看exec系列函数. exec系列函数可以把某个进程导入到调用exec系列函数的进程中,并且把当前进程替换成一个新的进程,在前面的章回中我们也介绍了,执行exec系列函数会触发写时复制,因此新进程拥有自己的代码区

替换进程映像 - exec系列函数

exec系列函数由一组相关的函数组成,它们在进程的启动方式和程序参数的表达方式上各有不同.exec函数可以把当前进程替换为一个新进程,新进程由patch或file参数指定.你可以使用exec函数将程序的执行从一个程序切换到另一个程序.例如,你可以在启动另一个有着受限使用策略的程序前,检查用户的凭证.exec函数比system函数更有效,因为在新的程序启动后,原来的程序就不再运行了. 替换进程映像 - exec系列函数

linux的子进程调用exec( )系列函数

exec( )函数族 : 以下我们来看看一个进程怎样来启动还有一个程序的运行.在Linux中要使用exec函数族.系统调用execve()对当前进程进行替换,替换者为一个指定的程序,其參数包含文件名称(filename).參数列表(argv)以及环境变量(envp).exec函数族当然不止一个.但它们大致同样,在Linux中,它们各自是:execl,execlp.execle,execv,execve和execvp,以下我仅仅以execlp为例,其他函数到底与execlp有何差别,请通过mane

轻松理解execl系列函数

execl函数功能如下:启动一个可执行文件,并且对他进行传送参数.一些原型如下 #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

Linux中exec()执行文件系列函数的使用说明

函数原型: 描述:    exec()系列函数使用新的进程映像替换当前进程映像.    工作方式没有什么差别, 只是参数传递的方式不同罢了. 说明:    1. 这6个函数可分为两大类: execl()系列 和 execv()系列.    2. `l' 是指把所有传递给程序的参数依次列(list)出来.        `v' 是指把所有参数放到容器(数组, vector)中再一次性传入.         不论是list出来还是vector them all, 参数的最后一个都必须为空指针((ch

execve 系列函数

exec系列函数(execl.execlp.execle.execv.execvp) 1带 p 的exec函数:execlp,execvp,表示第一个参数path不用输入完整路径,只有给出命令名即可,它会在环境变量PATH当中查找命令 2 execl* 和execv*的区别是,前者一一列出参数,后者用指针数组这出. 3 最后有e的表示自己制定环境变量,不带e的表示使用进程环境原有的环境变量.

linux tricks 之VA系列函数.

VA函数(variable argument function),参数个数可变函数,又称可变参数函数.C/C++编程中,系统提供给编程人员的va函数很少.*printf()/*scanf()系列函数,用于输入输出时格式化字符串:exec*()系列函数,用于在程序中执行外部文件(main(int argc, char* argv[]算不算呢,与其说main()也是一个可变参数函数,倒不如说它是exec*()经过封装后的具备特殊功能和意义的函数,至少在原理这一级上有很多相似之处).由于参数个数的不确

【转载】linux进程控制-exec系列 exec系统调用

inux进程控制-exec系列 说是exec系统调用,实际上在Linux中,并不存在一个exec()的函数形式,exec指的是一组函数,一共有6个,分别是: #include <unistd.h>int execl(const char *path, const char *arg, ...);int execlp(const char *file, const char *arg, ...);int execle(const char *path, const char *arg, ...,