您的位置:首页 > 运维架构 > Shell

使用exec函数组调用执行shell脚本

2014-08-14 10:53 134 查看
linux下的exec函数不是单一的函数,而是一个函数组,分别为:

[cpp] view
plaincopy

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脚本:

[plain] view
plaincopy

#!/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.程序

[cpp] view
plaincopy

#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;

}

运行结果如下:

[cpp] view
plaincopy

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函数组介绍

/article/7609839.html

以及APUE 8.10

本人享有博客文章的版权,转载请标明出处http://blog.csdn.net/baidu20008
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: