对于会shell编程的人肯定不会对#!/bin/sh感到陌生。这一行字符串到底意味着什么呢?
应该明确,所谓解释器就是指#!行后面的可执行程序。
我们从exec函数说起。
exec函数总共包括六个函数。
#include <unistd.h>
int execl(const char *pathname,const char *arg0,.../*(char *)0 */);
int execv(const char *pathname,char *const argv[]);
int execle(const char *pathname,const char *arg0,.../*(char *)0,char *const envp[] */);
int execve(const char *pathname,char *const argv[],char *const envp[]);
int execlp(const char *filename,const char *arg0,.../*(char *)0 */);
int execvp(const char *filename,char *const argv[]);
这些函数作用都是一样:执行一段新的代码。
其中execl函数,第一个参数path是设置了执行位的可执行文件的路径,后面的可变参数列表分别指向了传递给此执行文件的参数列表,(包括了参数0,即是执行文件的名称),最后一个参数(char *)0则表示参数列表结束。
对于解释器,exec函数(以execl为例)是这样执行的,如果path是指向了一个脚本,脚本的第一行是以#!开头,则这样调用:
以#!后面的字符串为命令,后面加上execl参数列表指定的参数,这样就形成了新的程序的执行。
以下是unix高级环境编程中关于解释器文件的例子。
首先我们在/home/qiqijianglu下创建一个解释器文本:
文本的名字为testinterp
文本里的内容如下:
#!/home/qiqijianglu/echoarg foo
----------------------------------------------------------------------------
分隔线及分隔线以下不是文本里的内容,echoarg程序内容如下:
#include "apue.h"
int main(int argc,char *argv[])
{
int i;
for(i=0;i<argc;i++)
printf("argv[%d]: %s\n",i,argv[i]);
exit(0);
}
-------------------------------------------------------------------------------
那么echoarg就是一个解释器,它回送每一个命令行参数。
下面我们要编写一个执行一个解释器文件的程序,文件名为197.c,代码如下:
#include "apue.h"
#include <sys/wait.h>
int main(void)
{
pid_t pid;
if((pid=fork())<0)
err_sys("fork error");
else if(pid==0)
{
if(execl("/home/qiqijianglu/testinterp","testinterp","myarg1","MY ARG2",(char *)0)<0)
err_sys("execl error");
}
if(waitpid(pid,NULL,0)<0)
err_sys("waitpid error");
exit(0);
}
---------------------------------------------------------------------------------------------------
这样我们用命令cc 197.c编译197.c,然后运行该程序./a.out
argv[0]: /home/qiqijianglu/echoarg
argv[1]: foo
argv[2]: /home/qiqijianglu/testinterp
argv[3]: myargl
argv[4]: MY ARG2
execl是这样执行命令的:
/home/qiqijianglu/testinterp的内容是#!/home/qiqijianglu/echoarg foo则#!后面的字符串/home/qiqijianglu/echoarg foo加上命令行参数列表组成了新的程序行/home/qiqijianglu/echoarg foo /home/qiqijianglu/testinterp myargl MY ARG2
注意,内核取execl调用中的pathname而非第一个参数testinterp,因为一般而言,pathname包含了比第一个参数更多的信息。
对于testinterp脚本,我们在shell中调用它时,shell会调用fork,exec,wait来执行它,首先,exec函数对#!行分析后得出此脚本的解释器为/home/qiqijianglu/echoarg,然后就把命令行处理成了/home/qiqijianglu/echoarg foo ./testinterp