Perl可执行文件_井叹号(#!)pound-bang的作用_可执行脚本的本质
一直以来,都没有搞清楚脚本中总是使用的井叹号(#!)的作用,现在学习perl,终于明白。
原理:
在支持#!的系统中,当敲入命令(可执行文件的文件名)时,Unix系统检查文件的前两个字节。如果这两个字节是#!,系统将#!后的部分作为执行器执行此命令(即执行这个可执行文件)。
简而言之,#!告诉sh,请用#!后面的命令执行本文件。
例如一个可执行文件abc.pl,第一行为:#! /usr/local/bin/perl -w,则在命令行输入命令abc.pl时,sh回去查看abc.pl的前两个字符,如果是#!,则执行如下命令 /usr/local/bin/perl -w abc.pl 。
再说的仔细一些,输入的命令一定是先给外壳程序shell去执行,当碰见#!时就改变执行者,如果没有碰到#!则仍用shell继续执行。(这个设计类似于CSS的Hack技术)
明白了原理,不难明白为啥写的很多shell脚本都要写上#! /usr/local/bin/zsh(等类似)。同时也知道怎么直接执行一个perl文件而不需要每回都多写一个perl命令了。
不支持#!符号的Unix系统怎么办:
注意此处说的仍是Unix系统系,但不支持#!符号。
如果不支持#!,也就是说没法改变执行者,仍必须由外壳程序shell来执行到底,此时需要用到shell的特性:eval和exec。
此处仍拿perl程序abc.pl来说。
#! /usr/local/bin/perl eval "exec /usr/local/bin/perl -s $0 $*" if 0; print "Perl Message! \n";
分析一下执行过程
1. 首先shell会逐行执行,由于此处说的Unix不识别#!号,但识别注释符号#,那么第一行就被当做注释掠过去了。
2. 接下来由shell执行eval和exec,用perl命令来执行敲入的命令行和敲入的参数($0和$*)。注意这里发生了执行转换。
3. perl执行此文件abc.pl时,也把带#!号这行作为注释对待,执行第二行和第三行时由于有if 0 这个总是为false的判断,因此也不执行这两行。接下来就执行正常处理了。
(这个设计真的很棒,有些回调的味道,同时又类似于CSS的Hack技术)
-s作为开关参数传给perl,让perl从标准的标准$PATH中寻找文件(abc.pl),从而避免了写abc.pl的绝对路径。
这个设计执行时有shell作为父进程,perl执行作为子进程,两个进程都执行了同一个文件abc.pl。
留一个问题:shell父进程怎么执行print "Perl Message! \n";及后面的行?
DOS/Windows下怎么让perl程序成为可执行的
也就是说,使用abc.pl时,键入abc arg1 arg2 而不是键入perl abc.pl arg1 arg2 或者键入c:/setup/perl/bin/perl abc.pl arg1 arg2
首先windows/DOS下,仅仅可以执行exe/msi/bat/com程序。道理还是一样的,用hack技术了。
同时windows下的perl有一个perl.exe可以帮助执行。
并不常用,略。