linux下的exec函数不是单一的函数,而是一个函数组,分别为:
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[]);
可以看到这个函数组的函数名都是有exec和添加l、v、p、e这四个字母的组合而来。
字母l代表list,表示该函数取一个参数表,即要求将新程序的每个命令行参数都说明为一个单独的参数,必须以(char*)0结尾;
字母v代表vector,表示该函数取一个argv[]矢量,即要求先构造一个指向各个参数的数组指针,然后将该数组地址作为这个exec函数的参数;
字母p代表path,表示该函数取filename作为参数,并且使用PATH环境变量寻找可执行文件;
字母e代表environ,表示该函数取一个envp[]数组,而不使用当前环境。
这里,字母l和v互斥,即不能同时出现在exec函数中。
exec函数最为常见的是调用替换执行可执行文件,诸如:
execl("/bin/echo","echo", "executed by execl", NULL)中的echo;
execl("/bin/ls", "ls","/azuo", "-la", (char *)0 )中的ls;
execlp("echo", "echo","executed by execlp", NULL)中的echo;
但是另外一种执行方式是调用执行shell脚本。
有两种方法使用exec函数组调用shell脚本:
方法一:使用execl函数,这种方法与execl("/bin/echo", "echo", "executed byexecl", NULL)中的echo 类似。它直接使用可执行文件来执行脚本。
将/bin/echo换为了/bin/sh或/bin/bash。
这样我们可以使用如下函数调用t.sh脚本
execle("/bin/sh","sh","t.sh",NULL,NULL);
可以向t.sh脚本传入参数,这里省略。
方法二:使用execle函数
char* env[]={"envargv=\"i'm inenvironment! you see me.\"",NULL};
execle("t.sh","nouse","i'min argv!",NULL,env);
使用execle函数,也可以向shell脚本传递参数,在脚本中使用$0,$1$2等等来进行引用,$1为第一个参数,这个与main函数的argv类似。
这里execle函数发现pathname参数并不是由连接编辑器产生的机器可执行文件,则认为该文件是一个shell脚本,于是试图调用/bin/sh,并以filename作shell的输入。与上一个方法不同的是,execl函数运行shell脚本即使用可执行文件,不要求t.sh有执行权限。而execle函数则要求使用者拥有被调用的shell脚本的执行权限,否则会调用失败,且通常不会提示Permission denied错误信息。所以在使用execle调用shell脚本时要确保脚本自身拥有执行权限。
这样的差别和终端下类似:
终端下使用sh t.sh 不需要执行权限而./t.sh就需要了。
以下为一组具体实例:
t.sh脚本:
#!/bin/bash
if [ -n "$envargv" ]; then
echo "you can use the var of env."
echo "the vat in env:$envargv"
echo "the var in argv:$1"
fi
echo "er,you see me again."
测试主程序:
tryexec.程序
#include <unistd.h>
#include<string.h>
#include <stdio.h>
int main(int argc,char** argv)
{
char par[10]={0};
strcpy(par,argv[1]);
if(!strcmp(par,"execl"))
{
printf("******************execl*****************\n");
execl( "/bin/cat","cat","/home/gs/exec.txt",NULL);
}
if(!strcmp(par,"execle1"))
{
printf("**************call shell script (1)*****************\n");
execle( "/bin/sh","sh","t.sh",NULL,NULL);
}
if(!strcmp(par,"execle2"))
{
printf("**************call shell script(2)*****************\n");
char* env[]={"envargv=\"i'm in environment! you see me.\"",NULL};
execle("t.sh","nouse","i'm in argv!",NULL,env);
}
return 0;
}
运行结果如下:
root@ubuntu:/home/gs/linuxapiusage# gcc tryexec.c
root@ubuntu:/home/gs/linuxapiusage# ./a.out execl
******************execl*****************
hello,you see me.
root@ubuntu:/home/gs/linuxapiusage# ./a.out execle1
**************call shell script (1)*****************
er,you see me again.
root@ubuntu:/home/gs/linuxapiusage# ./a.out execle2 此处没有t.sh的执行权限
**************call shell script(2)*****************
root@ubuntu:/home/gs/linuxapiusage# ./a.out execle2 增加执行权限 chmod +x t.sh
**************call shell script(2)*****************
you can use the var of env.
the vat in env:"i'm in environment! you see me."
the var in argv:i'm in argv!
er,you see me again.
以上为execl和execle来调用的shell脚本,当然使用execv 和 execve也可以,只不过是参数传入格式的问题了。
更多信息请参考:
exec相关
http://man7.org/linux/man-pages/man3/exec.3.html
或linux下man exec
exec函数组介绍
http://blog.csdn.net/fisher_jiang/article/details/5608399
以及APUE 8.10
本人享有博客文章的版权,转载请标明出处http://blog.csdn.net/baidu20008