引言 #include 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[]); exec族函数一共有上面所列的5个,作用都是一样:执行一段新的代码。区别只是向函数传递的参数方式不同而已,其中execl函数:第一个参数path是指向设置了执行位文件的路径,后面的可变参数列表分别指向了传递给此执行文件的参数列表(包括了参数0,即是执行文件的名称)。最后一个参数为(char *) 0,表示参数列表结束。 对于解释器,exec族函数是这样做的(以execl为例),如果path是指向了一个脚本,脚本的第一行以#!开头,则这样调用: 以#!后面的字符串为命令,后面加上execl参数列表中指定的参数列表,这样形成了新的程序执行。 下面我们以例子来验证这个结果: 下面这个C程序的作用是回射所有命令行参数。 代码: QUOTE: /* Program source : showargs.c * * Program name : showargs */ #include int main(int argc, char *argv[]) { int i; for(i = 0; i < argc; i++) { printf("arg[%d]: %s\n", i, argv); } return 0; }编译:gcc -o showargs showargs.c 执行: 代码:QUOTE: $ pwd /home/kiron $ ./showargs arg1 arg2 arg[0]: ./showargs arg[1]: arg1 arg[2]: arg2 我们在同一个目录下再写一个脚本: #!/home/kiron/showargs addargs 这个脚本就只有一行,这个脚本我们命名为testexec,加上执行位后,执行情况如下: $ ./testexec arg[0]: /home/kiron/showargs arg[1]: addargs arg[2]: ./testexec 可能会有人对第2个参数./testexec不理解,再引出一个C程序: /* Program source : mytest.c * * Program name : mytest */ #include int main(void) { execl("/home/kiron/testexec", "testexec", "arg1", "arg2", (char *) 0); return 0; }编译:gcc -o mytest mytest.c 执行: 代码: $ ./mytest arg[0]: /home/kiron/showargs arg[1]: addargs arg[2]: /home/kiron/testexec arg[3]: arg1 arg[4]: arg2 仔细观察上面的三个例子,正如在开始时讲到的,exec族函数的处理是把#!后面的字符串为命令,后面加上execl参数列表中指定的参数列表,这样形成了新的程序执行。分析一下mytest.c源程序,execl把命令的结果是这样执行的: /home/kiron/testexec的内容是#!/home/kiron/showargs addargs,则#!后面的字符串"/home/kiron/showargs addargs"加上命令参数列表:"/home/kiron/testexec arg1 arg2"就形成了新的程序行:/home/kiron/showargs addargs /home/kiron/testexec arg1 arg2。 对于testexec脚本,我们在shell中调用它时,shell调用了fork,exec,wait来执行它,也就是和程序mytest.c一样用了exec函数,首先,exec函数对#!行分析后得出此脚本的解释器为/home/kiron/showargs,然后就形成了把命令行处理成了:“/home/kiron/showargs addargs ./testexec”。
#!/bin/sed -f s/UNIX/unix/p 执行./mysed时出错了。因为被解释成了"/bin/sed -f ./mysed",其中-f选项是表示以文件里的内容作为sed的命令输入,但sed的命令输入不能对"#!/bin/sed -f"解释,那么程序出错了。
#!/usr/bin/awk -f BEGIN { sum = 0; } {sum += NF;} END { printf("file \"%s\" have %d line, %d words.\n", FILENAME, NR, sum); } 设置执行位之后,执行如下: |