linux解释器原理

对于会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

你可能感兴趣的:(linux解释器原理)