软件测试


网站首页 | 软件测试论坛 | 软件测试培训 | 软件测试博客 | 软件测试杂志 | 软件测试沙龙 | 软件测试下载 | 软件测试顾问
业界新闻 | 软件测试人才 | 软件测试技术 | 软件测试工具 | 行业软件测试 | 软件测试管理 | 软件质量专栏 | 软件开发专栏
当前位置:首页>>软件开发专栏>>操作系统>>正文
Linux解释器原理
文章出处:网络 作者: 发布时间:2006-11-20

引言
    使用Shell进行工作的人们对Unix/Linux下的Shell编程都很熟悉,在所有的Shell编程的书中都会提到#!/bin/bash,而这里到底包含了些什么?对操作系统而言,这一行字符串意味着什么?但首先要我们明确一下,所谓解释器就是指#!行后面的可执行的程序。



一、我们从exec族函数谈起
    如果你从不写C程序,可能需要对本节的内容看得更为仔细并且试验一下。

代码:

 

#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

我们在同一个目录下再写一个脚本:

代码:QUOTE:

#!/home/kiron/showargs addargs

    这个脚本就只有一行,这个脚本我们命名为testexec,加上执行位后,执行情况如下:

代码:QUOTE:

$ ./testexec
arg[0]: /home/kiron/showargs
arg[1]: addargs
arg[2]: ./testexec

    可能会有人对第2个参数./testexec不理解,再引出一个C程序:

代码:QUOTE:

/* 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”。

注意:#!行中的解释器的路径必须是全路径,exec函数并不对其特殊处理,比如用PATH变量来搜索它的真实路径,所以路径是由程序员来保证正确的。

二、我的脚本第一句必须得是#!/bin/bash吗?
    不必。通过上面的解释,其实第一句的#!是对脚本的解释器程序路径,脚本的内容是由解释器解释的,我们可以用各种各样的解释器来写对应的脚本,比如说/bin/csh脚本,/bin/perl脚本,/bin/awk脚本,/bin/sed脚本,甚至/bin/echo等等。下面是一个例子:

代码:
#!/bin/echo -e
    这里只有一行的程序(实际上它也只能是一行,echo程序并不是被设计成像awk那样的编程语言,能写成源程序文件)命名为myecho,加上权限后执行它:

代码:
$ ./myecho "hi\a"
./myecho hi
如果你的echo支持-e选项并且你工作的环境还算安静,你在得到上面的结果的时候也应该听到清脆的终端响铃。


三、我能利用解释器来做什么?
    但是上面的echo脚本实际应用时并没有什么作用,我们可以得出一个小小的实验结果,并不是所有的可执行二进制文件都可以用来写解释器脚本。如果你有一个可编程的解释器,那你或许能编写该解释器的程序来简化你工作。比如说常用到的解释器如awk,perl,bash等等。但是正如我们上面总结的实验结果,并不是全部的可编程程序都是有用的解释器,exec脚本时,能从第一行得到脚本的解释器,然后用exec去解释脚本,也包括了形如#!/PATH/的第一行,如果该解释器对这行不能忽略的话,就会出错,另外解释器也必须要对余下的程序语句能解释(例如:上面myecho程序加一些"hello world"的行来,会有效吗?下面的mysed程序中的s/UNIX/unix/p也是一样的道理)。像awk,perl,bash等程序对#开头的行当成注释行处理,就能写成有用的脚本。
再看下面的mysed程序,

代码:QUOTE:

#!/bin/sed -f
s/UNIX/unix/p

    执行./mysed时出错了。因为被解释成了"/bin/sed -f ./mysed",其中-f选项是表示以文件里的内容作为sed的命令输入,但sed的命令输入不能对"#!/bin/sed -f"解释,那么程序出错了。


    所以,有用的解释器应该是类似bash,perl,awk的程序,并且能对一些规定的语句有解释功能的。下面给出一个awk程序写的统计文件行数和单词数的脚本程序myawk。

代码:QUOTE:

#!/usr/bin/awk -f
BEGIN {
  sum = 0;
}
{sum += NF;}
END {
  printf("file \"%s\" have %d line, %d words.\n", FILENAME, NR, sum);
}

设置执行位之后,执行如下:

代码:
$ echo -e "hi\nhello world">test.txt
$ ./myawk test.txt
file "test.txt" have 2 line, 3 words
    这里执行./myawk被执行成“/usr/bin/awk -f ./myawk test.txt”,因为awk的命令中,以#开头的行被认为是注释行而忽略,awk忽略了第一行"#!/usr/bin/awk -f",正确的以非#开头行当成模式和命令的输入并能对其解释,所以这个程序是正确的,能被顺利地执行。

    关于Linux系统的解释器的介绍就说到这了,希望大家能对解释器的原理有更多的认识。


站内搜索
相关文章
◎自动安装Windows
◎Unix系统安全必读(2)
◎Unix系统安全必读(1)
◎GCC 命令行详解
◎linux shell 脚本入门
◎Linux网络安全之经验谈(2)
◎如何在Linux下搭建Web应用环境
◎Linux网络安全之经验谈(1)
◎linux下的网络分析Tcpdump命令的使用
◎linux下网络经典问题指南篇
◎资深Linux程序员的开发经验谈
◎Linux操作系统12则经典应用技巧
◎Linux上的C/C++编译器gcc/egcs详解
◎Linux无线网络技术概述:WLAN
◎Unix系统安全必读
◎Linux内核模块和驱动的编写
◎Linux 下 C++程序的异常处理技巧
◎如何在Linux环境模拟DOS命令
◎Linux的常用网络命令
◎操作系统探测
◎实时操作系统VxWorks在跟踪雷达系统中的应用
◎Linux嵌入式系统开发平台选型探讨
◎uCOS和uClinux的比较
◎XP自带网络诊断程序解决网络故障
◎完全掌握Windows系统还原12招
◎Linux 下 C++程序的异常处理技巧
◎巧用DLL文件来隐藏你的资料
◎使用 Bash shell 脚本进行功能测试
◎学会使用Linux性能分析工具
◎Qmail安装及设定
◎Microsoft 群集服务 (MSCS) 简介
◎Linux初学之学习方向和方法浅谈
◎Linux常见紧急情况处理方法
◎Linux操作系统中的7件武器详解
◎Windows XP鲜为人知的N招
◎嵌入式系统Boot Loader 技术内幕(4)
◎嵌入式系统Boot Loader技术内幕(3)
◎嵌入式系统Boot Loader技术内幕(2)
◎嵌入式系统Boot Loader 技术内幕(1)
◎Unix发展趋势:向开放架构迈进
◎Linux下硬盘分区详解
◎安全高效 方便简洁—《Sun Solaris 10》测试手记
◎linux使用大全(9)
◎linux使用大全(8)
◎linux使用大全(7)
◎linux使用大全(6)
◎linux使用大全(5)
◎linux指令大全(4)
◎linux指令大全(3)
◎linux指令大全(2)
热门文章
◎Sun Solaris指令大全
◎vi使用手册
◎linux shell 脚本入门
◎学会使用Linux性能分析工具
◎Linux指令大全(1)
◎嵌入式系统Boot Loader技术内幕(2)
◎Linux操作系统12则经典应用技巧
◎Linux的常用网络命令
◎Linux上的C/C++编译器gcc/egcs详解
◎Windows XP鲜为人知的N招
◎Linux下硬盘分区详解
◎嵌入式系统Boot Loader 技术内幕(1)
◎Qmail安装及设定
◎linux指令大全(2)
◎uCOS和uClinux的比较
◎Linux嵌入式系统开发平台选型探讨
◎嵌入式系统Boot Loader技术内幕(3)
◎GCC 命令行详解
◎linux指令大全(3)
◎Linux内核模块和驱动的编写
◎linux指令大全(4)
◎使用 Bash shell 脚本进行功能测试
◎linux下网络经典问题指南篇
◎linux使用大全(5)
◎Linux初学之学习方向和方法浅谈
◎嵌入式系统Boot Loader 技术内幕(4)
◎linux使用大全(9)
◎linux命令全记录
◎linux使用大全(6)
◎linux使用大全(7)
◎巧用DLL文件来隐藏你的资料
◎安全高效 方便简洁—《Sun Solaris 10》测试手记
◎linux使用大全(8)
◎Linux无线网络技术概述:WLAN
◎Linux操作系统中的7件武器详解
◎如何在Linux下搭建Web应用环境
◎XP自带网络诊断程序解决网络故障
◎完全掌握Windows系统还原12招
◎实时操作系统VxWorks在跟踪雷达系统中的应用
◎自动安装Windows
◎Linux常见紧急情况处理方法
◎操作系统探测
◎资深Linux程序员的开发经验谈
◎如何在Linux环境模拟DOS命令
◎Unix系统安全必读
◎Unix系统安全必读(1)
◎Linux 下 C++程序的异常处理技巧
◎linux下的网络分析Tcpdump命令的使用
◎linux资源管理(1)
◎Linux 下 C++程序的异常处理技巧

Google提供的广告