A!die Software Studio Welcome to A!Die Software Studio

Perl 语言全面编译

by 天水-S.Tanshuai
2011-05-29 09:58:30
简 述 

本文将详细讲述Perl的编译方法,献给所有热爱、喜欢Perl的程序员们。 
Perl自从面世以来1.0版本到现今的5.6版本,一直都有编译程序,主要因为国内的中文资料很少,大多数人不愿意去看或者不懂得英文资料,所造成不知道器编译方法。即使是很多Perl界高手也同样有此类问题。Perl编译方法五花八门,各种编译方法都有其重要的意义和弱点。另一方面Perl编译方法不能流行的原因是,本身Perl就是一个免费的东西,人们不希望Perl成为编译的商品,但是在国内也是因此而拖累了Perl的发展步伐。但在此我不赞成也不推崇Perl程序的编译,Perl编译有小些局限性,但是仍然可以完成所有任务,想要达到良好的编译效果,需要高超的编程技术和相关经验,重要的是对OOP(面向对象的程序设计)的了解,将会使得你的Perl程序更加易于编译,运行速度更快,兼容性更广等特性。 
以前我写过Perl在可嵌入式技术方面技术文章。它的优势和其它嵌入语言无法比拟的兼容性,Perl不但拥有PHP的可嵌入HTML技术,也同样支持用PerlScript写ASP的。但是如果你希望你的程序可以编译执行,那么可嵌入式方法显然是不可能的。我几乎不用ePerl、mod_perl等可嵌入式Perl HTML 页,但是我更不赞成很多人把HTML置入程序之中,这两种方法都有其好处以及坏处。我推崇模板方式的编写方法,大家可能也用过模板方式,可能认为它在页面量处理方面有很多问题?但是,那些都是陈旧古老的方式,也是说明你并未精通Perl语言,采用模板方式调入HTML页是相当好的方法,几乎可以达到所有可嵌入式技术的功效,也可以像HTML程序内置方式的灵活操纵性。我觉得程序员和HTML制作员是不同的,如果我们采用ePerl、PHP、ASP,那么你就不是一个真正的程序员,那只是HTML技术的服务器处理部分罢了,真正的程序是程序本身,而不附带任何其它特性。 
我认为好的教学文章,应该让读者充分了解内容,充分扩展层面。诸如编写一个Httpd程序,有很多传统的程序员根本不了解 http的通讯协议,即使讲了很多内容,但是仍然搞得半懂不懂。本文将会充分扩展层面,让读者了解更多的技术资料,而不必看完本文后又要去寻找关联技术资料。同样国内目前有很多技术性书籍,都是来自国外的译本,但是很多译者并非此技术专家,在翻译的时候很多东西无法充分理解,带来的时间上的障碍。我希望国内的编程专家能够写一些有用的技术文章和书籍,因为我看过很多国人自己写的文章都容易理解和操作。但是问题在于都偏向与基础教学,目前急切地需要有更深层次的技术资料。 
  
内容大纲: 
1) PerlApp和PerlSvc编译方法 New! Easy! 
2) Perl2Exe 编译方法 
3) PerlCC 编译方法 
4) PerlCC之Bytecode 编译解析法——Just Like Java Program!  New! Cool! 
5) OOP面向对象的程序之为编译而设计 
6) HTML模板编程方式——真正的WEB程序(Program)  Good! 
7) 联合编译以及实例  Advanced! 
  
说明:如何选择阅读以上内容是很重要的,以上内容并非适合各个阶层的Perl程序员。PerlApp和PerlSvc适合在Windows2000环境下编程初学者和一般的Perl程序设计人员,Perl2Exe适合在非Windows和Windows95/98/Me 环境下编程初学者和一般的Perl程序设计人员。PerlCC适合与任何操作系统平台,但是操作复杂,适合于中级程序员和高级程序员开发大宗商业化软件(公众客户)使用。ByteCode是一种新型的编译方式,类似Java,它需要Perl解析器的支持,但是它是灵活性最高的编译方式,适合中级程序员和高级程序员开发大宗商业化软件(服务商)使用。如果你希望你可以编写出一个出色的Perl编译的程序,那么你必须阅读第4节,它将告诉你如何使用面向对象的程序设计技术来实现Perl编译程序的高效良好的开发环境和模式。 
  
第一节 PerlApp和PerlSvc编译方法 

PerlApp和PerlSvc是ActiveState 公司开发的,它属于 Active Perl Dev Kit(PDK)产品。本编译方法只适合于Windows2000上运行,其它系统均无法正常使用,编译程序必须是标准Perl和ActivePerl。 
PDK下载地址:http://ftp.tanshuai.net/pub/  ftp://ftp.tanshuai.net/pub/ 
PerlApp和PerlSvc,前者是标准的应用程序,后者是Windows2000的服务程序(类似与IIS,一开机就启动的服务程序,而且无法中断它的运行)。他们有两种运作模式:依靠(Dependent)和独立(Freestanding),“依靠”模式程序运行的系统上必须有Perl解析器和相关模块,这样的程序相对较小;“独立”模式,Perl解析器等相关模块都会完全嵌入在程序之中,这样的程序在任何Windows2000操作系统上都可以顺利运行,而不需要额外的支持,但是程序相对较大。 
使用方法: 
标准使用方法(“依靠”模式): 
perlapp <脚本名> 
这样程序就会创建一个以脚本名命名的可执行文件<脚本名.exe> 
“独立”模式: 
perlapp(或者perlsvc) –f <程序名> 
定义输出可执行文件名: 
perlapp(或者perlsvc) –e=tanshuai.exe test.pl 
它将会把test.pl文件输出的可执行文件名改为“tanshuai.exe”。 
设置程序属性: 
perlapp(或者perlsvc) -i=<类表> <程序名> 
类表名 目标项目 
Filenumber 文件号码 
Productnumber 产品号码 
Productname 产品名称 
Legaltrademarks 合法商标 
Filedescription 文件说明 
Originalfilename 原文件名 
Fileversion 文件版本 
Comments 注解 
Productversion 产品版本 
Companyname 公司名称 
Internalname 内部名称 
Legalcopyright 版权 

这个时候有些人可能不大明白,这个是干什么用的。如果你曾经编写过Win32程序,那就会知道,它是Windows程序的版本说明(如图1)。 



图1 Perl.exe文件的版本说明 
名称与数值用“;”分开。而且所有项目值都需小写。 
清理PerlCtrl 的DLL: 
perlapp(或者perlsvc) –c <程序名> 
添加模块: 
perl(或者perlsvc) –a=<列表> 
如:perlapp tanshuai.pl –a=IO:Socket;XML::Parser;Tanshuai::Http;MP3; 
这样模块IO:Socket,XML::Parser,Tanshuai::Http和MP3就被置入程序内。 
Perl图形界面: 
perlapp(或者perlsvc) –g <程序名> 
如果你的程序非命令行或者CGI,是T/K图形界面的话,就需要采取这个命令。 
排除 Perl56.dll: 
perlapp(或者perlsvc) –x <程序名> 
Perl56.dll是PerlApp执行的关键,但是如果你不希望他和你的程序在一起,你可以把它排除,另行安置,但是主意,一定要保证它的存在否则就无法正确运行 
添加额外文件: 
perlapp(或者perlsvc) –b=<文件列表> <程序名> 
如果你希望在程序内部打开文件,请使用这个命令。 
如:open(FILE,“./PerlAPP.TXT“);@FILE=;close(FILE); 
这样就必须打开“PerlAPP.TXT“文件,但是你如果把它置入程序,它将会在内存中打开。(无法写入) 
报告嵌入模块错误: 
perlapp(或者perlsvc) -r <程序名> 
一些模块无法嵌入,使用该命令可以得出相关信息。 
输出详细信息: 
perlapp(或者perlsvc) <程序名> -v 
如:perlapp tanshuai.pl –v 
输出: 
Using myScript.pl for script name 
Input script name: tanshuai.pl 
Output exe name: tanshuai.exe 
Exe Mode: Perl Dependent 
Creating dependent executable 
  
解释:PerlApp 和PerlSvc无法在Windows95/98/ME PerlApp使用的部分Win32 API函数未被支持。 



第二节 Perl2EXE 编译方法 

Perl2EXE 可以在大多数流行系统上编译运行,但是我几乎不用它,我认为它是“最低级”编译。而且它也是最容易被反编译的程序。所以我不推崇它,也不愿意用它。不过适合很多初学者。 
它的原理很简单,知识把原来的Perl代码放入程序中和内置的解析其共同运行,而且速度不如PerlAPP。 
Perl2EXE 同样可以在 http://ftp.tanshuai.net/pub 和 ftp://ftp.tanshuai.net/pub/ 下载。 
标准方法: 
perl2exe <程序名> 
Perl解析器选项值设定: 
perl2exe –perloption=“<参数>“ <程序名> 
参数主要就是perl解析器的参数如:-w –X –e 等等。 
共享dll库: 
perl2exe –small <程序名> 
如果你是多个程序编译,那么使用这个命令,比较“划算“,你只要把它们的共享dll库,复制到共同的执行目录下,即可。共享DLL库:p2xdll.dll或者p2x560.dll。 
启动图形界面: 
perl2exe –gui <程序名> 
和perlapp是同样的作用。 
设置执行程序的图标: 
perl2exe –icon=<图标文件名> <程序名> 
设置输出文件名: 
perl2exe <程序名> -o=<文件名> 
设置运行系统平台: 
perl2exe –platform=<系统名称> <程序名> 
如:Sun操作系统 perl2exe –platform=sun program.pl Linux操作系统 perl2exe –platform=linux program.pl 

第三节 PerlCC 编译方法 

PerlCC是Perl的最好最优秀最强的得编译器,而且是免费的。但是它的调试与运作是比较方“烦”人的。特别是在微软的Windows就更令人头疼。 
PerlCC编译器的原理是分析Perl原代码,然后根据标准转换方式,转换成C语言,当然这里的C全部采用Perl的头文件(Header),也就是全部采用Perl的函数,即使你只有一行的 “print “hello world”;”都需要无数行的定义后才会出现这样的效果。但是令人惊奇的是perl编译后的这个“hello world”比C/C++的编译后的可执行文件还要小。采用PerlCC转换出来的C源代码几乎是不可读(不可理解)的,几乎比汇编语言还令人费解。所以这样的程序即使被反编译出来,它的源代码也是会令人无法琢磨,但是这种程序根本几乎无法反编译,至少目前是,我相信只要Windows未被反编译那么它编译出来的Perl可执行程序也同样无法反编译。 
如果使用PerlCC是大家最关心的事情,在Unix-Style系统是,凡是安装perl5.0以上版本的都可以使用PerlCC,编译程序,但是必须有C编译器。这个我就不必太多说了。因为这个方法不大适合初学者,一般中级程序员对Unix-Style系统应该是较为了解的。 
在Windows中,一定要安装VC6.0(也可以是GCC,但是安装复杂)否则仍然无法编译,安装VC6.0是简单的事情,只要找到微软VC6的光盘,安装。 
然后,下载Perl源代码(地址:http://ftp.tanshuai.net/pub/ ),下来后解开压缩(Windows可以用Winzip)。 
UNIX-Style 命令行模式下: 
#cd <文件解压缩后的目录> 
#make 
#make test ~可选 
#make install ~完成安装 
#export PATH=$PATH;/<安装目的目录路径>; ~设置变量 
Windows 命令行(Command.com CMD.COM)模式下: 
C:\>cd <文件解压缩后的目录\win32> 
C: <文件解压缩后的目录\win32\> nmake 
C: <文件解压缩后的目录\win32\> nmake test ~可选 
C: <文件解压缩后的目录\win32\> nmake install 
Windows 95/98/Me 在 AutoExec.Bat文件中设置路径。 
Windows Nt/2000 在“控制面板”-〉“系统”-〉“高级”-〉“环境变量”中设置 
 注意:千万不要使用AtivePerl,而且最好在安装标准编译Perl后,删除AtivePerl,AtivePerl“不支持”PerlCC,虽然它也有带perlcc 但是至少我是永远都无法编译成功的,我也不知道为什么,我也不想知道为什么,因为很多程序是在Unix-Style 上运作的,大多数都是标准Perl,所以建议大家为了兼容所有操作系统,请尽量用标准Perl编写和解析程序。 
好啦,一切安装、设置就绪后,重新启动计算机后。我们进入我们想要编译的文件目录中,输入“perlcc <程序名>”(注意:这里的程序扩展名称必须是.pl .bat .p .pm,.cgi也不行,你可以修改perlcc.bat文件来支持其它扩展名)。 
输入以上命令后,会出现一大堆你可能看不懂的命令(这些你并不需要关心) 
例如我要编译一个内容为: 
print “ok”; 
的Perl程序,该文件名:abc.pl。 
输入: 
perlcc abc.pl 
PerlCC输出内容: 
------------------------------------------------------------------------------ 
Compiling abc.pl: 
------------------------------------------------------------------------------- 
  
Making C(abc.pl.c) for abc.pl! 
C:\perl\5.6.0\bin\MSWin32-x86\perl.exe -IC:/perl/5.6.0/lib/MSWin32-x86 -IC:/perl 
/5.6.0/lib -IC:/perl/site/5.6.0/lib/MSWin32-x86 -IC:/perl/site/5.6.0/lib -I. -MB 
::Stash -c abc.pl 
C:\perl\5.6.0\bin\MSWin32-x86\perl.exe -IC:/perl/5.6.0/lib/MSWin32-x86 -IC:/perl 
/5.6.0/lib -IC:/perl/site/5.6.0/lib/MSWin32-x86 -IC:/perl/site/5.6.0/lib -I. -MO 
=C,-l2000,-umain,-uattributes,-uDB,-uWin32 abc.pl 
Starting compile 
Walking tree 
Prescan 
Saving methods 
Bootstrap attributes abc.pl 
Writing output 
Loaded B 
Loaded IO 
Loaded Fcntl 
abc.pl syntax OK 
Compiling C(abc) for abc.pl! 
C:\perl\5.6.0\bin\MSWin32-x86\perl.exe -IC:/perl/5.6.0/lib/MSWin32-x86 -IC:/perl 
/5.6.0/lib -IC:/perl/site/5.6.0/lib/MSWin32-x86 -IC:/perl/site/5.6.0/lib -I. E:\ 
DOCUME~1\ADMINI~1\LOCALS~1\Temp/abc.pl.tst 
Couldn't open E:DOCUME~1ADMINI~1ocals~1temp/abc.pl.val 
cl -Od -MD -DNDEBUG -DWIN32 -D_CONSOLE -DNO_STRICT -DPERL_MSVCRT_READFIX -Od - 
MD -DNDEBUG -Ic:\perl\5.6.0\lib\MSWin32-x86/CORE -o abc abc.pl.c /link -nologo 
-nodefaultlib -release -libpath:"c:\perl\5.6.0\lib\MSWin32-x86\CORE" -machine: 
x86 -libpath:c:\perl\5.6.0\lib\MSWin32-x86/CORE c:\perl\5.6.0\lib\MSWin32-x86\CO 
RE\perl56.lib oldnames.lib kernel32.lib user32.lib gdi32.lib winspool.lib com 
dlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib netapi32.lib uuid.lib 
wsock32.lib mpr.lib winmm.lib version.lib odbc32.lib odbccp32.lib msvcrt.lib 
abc.pl.c 
然后再输入:abc.exe,如果输出结果与abc.pl一样,那么编译就成功了。这个程序是使用Visual C++的CL.EXE C/C++编译程序编译的。在Unix-Style下是使用CC或者GCC编译的。 
模块编译注意事项: 
目前PerlCC标准编译方式可以支持大多数程序的模块使用,但是我推荐尽量使用内部命令来完成相应任务,诸如IO::Socket模块可以使用socket 内部函数。因为 IO::Socket是无法在PerlCC下面编译成功的,为什么? 大家知道Perl很多的模块是本身Perl的语言和内部函数编写的。但是有一部分包括IO::Socket DBD DBI等这些常用的模块,由于Perl本身内置函数限制,采用了PerlXS接口通过C 程序达到目的的。这些是通过第三方程序达到目的模块是无法成功的编译。所以我建议尽量使用非含有第三方程序的模块编程。有人可能会说了,我使用了DBI DBD来操作数据库,难道让我放弃吗?不,我觉得任何事情都是有它的解决方法,Perl也一样。Perl Bytecode将会解决这一问题(详情情看第4节)。 
编程方式注意事项:我为什么要在开头说OOP 等相关技术以及在本文中提及了OOP的编写?因为PerlCC编译有一定的局限性,如果采用OOP就可以避免这个局限性,而且会更好的发挥,众所周知,OOP是编程发式的有一革命,你迟早都会涉及的,所以早一点总比晚一点好。我们经常编写程序的时候用“require”命令来引用其它Perl程序文件。然而这种方式不是PerlCC不支持,PerlCC当然支持,这个命令,但是问题在于它无法被编译入PerlCC的主程序内,也就是说主程序被编译了,然而外部引用的这个没有被编译,这样会造成很多问题,首先是暴露了原始代码,其次它人可以随意修改,肯能导致很多量(比如密码)被套出,也可以修改程序运行的模式。但是这也是有点,最后一节将会详尽讲述。 

第四节 PerlCC之Bytecode 编译解析法 

Bytecode 是 PerlCC的另一编译方法,必须在Perl5.6以后版本才有得支持。它的原理就好像Java一样,它会把Perl文件编译成二进制令人费解的乱码文件,它是采用类似MD5这样的反向加密编码,几乎不可能反编译,和可执行程序一样复杂,但是它不可以直接执行哦。想要执行它,必须用Perl解析器,就好像 Java 编译后必须有Java解析器,否则就无法执行。我习惯成为编译解析法,有的时候就说Just Like Java Progam! 
它的编译方法也不难,但是竟然有很多人都不知道,我问过很多Perl前辈,他们也不大了解这一方法。而且很多我也从来见过谁写过这样的程序(难道我是国内第一个知道的吗?:) 
使用方法:perlcc –b <程序名>  
编译后它会输出一个<程序名.plc>文件,你打开它看,定会吃惊。而且这种文件最小是180KB,比perlcc C语言转换编译多了很多。 
它的好处在于,一处编译到处使用。但是对于CGI就不大好处理。所以还是建议在各个平台进行编译。
例如我ByteCode编译上节的abc.pl程序文件: 
输入: 
perlcc –b abc.pl 
Perlcc –B 输出 
---------------------------------------------------------------------------- 
Compiling abc.pl: 
---------------------------------------------------------------------------- 
Making Bytecode(abc.plc) for abc.pl! 
C:\perl\5.6.0\bin\MSWin32-x86\perl.exe -IC:/perl/5.6.0/lib/MSWin32-x86 -IC:/perl 
/5.6.0/lib -IC:/perl/site/5.6.0/lib/MSWin32-x86 -IC:/perl/site/5.6.0/lib -I. -MB 
::Stash -c abc.pl 
C:\perl\5.6.0\bin\MSWin32-x86\perl.exe -IC:/perl/5.6.0/lib/MSWin32-x86 -IC:/perl 
/5.6.0/lib -IC:/perl/site/5.6.0/lib/MSWin32-x86 -IC:/perl/site/5.6.0/lib -I. -MO 
=Bytecode,-umain,-uattributes,-uDB,-uWin32 abc.pl 
abc.pl syntax OK 
好了,然后perl abc.plc 就可以执行了。 
执行注意事项: 
使用Bytecode 编译后的文件,你一般需要更名回原来的文件名,否则容易在运行程序后出现警告信息“Attempt to free unreferenced scalar.”虽然它对程序没有本质影响,但是不美观嘛,另一种解决方法就是使用 perl –X ,关闭所有警告消息,警告不等同与错误,所以一般情况下,某些警告是不必要的。 
同样ByteCode 编译程序可以被引用(require)但是不能调用(use),可以作为对象编程的对象。这是一个很灵活的东西,如果你希望你的模块被大家使用,但是不想让大家知道其中的操作,那么你就是用ByteCode,但是你的模块将永远不会被纳入CPAN。这种方法就好像OCX控件。 
但是注意,但是使用某个模块的时候,你必须保证使用该程序的机器上有这个模块,最简单的方法你可以把模块一起复制使用,但是有些第三方程序模块需要重新编译,你如果不希望其它人操作模块或者是看到引用的模块,也可以使用Bytecode。但是注意,一定要用require方法调用加密模块啊。这个世界总是这样,总会有些遗憾的,这样的话就不能用一些模块和OOP。 
不知道你了解Python这个语言否?它Perl很相像,比Perl还有简单呢。但是我认为很多东西都是抄Perl的,包括它得二进制编译方法,就和Perl Bytecode没有任何区别。反正大家也都知道PHP也是抄了Perl不少东西。 

第五节 OOP面向对象的程序之为编译而设计 

面向对象的程序设计已经不是什么新颖的话题和技术了。它在C++和Java中,尤为重要,哎,我觉得在写大宗程序的时候会很有帮助,但是在小程序里面反而麻烦,还不如普通的函数使用。OOP大多数基本的Perl教程都有说明,所以这里也不多讲“废话”,主要讲述OOP在编译Perl程序中的应用以及Perl OOP编写的技巧,所以值得一看。 
前面说过在PerlCC编译可执行程序的时候,不要使用require函数,这是没有错的。但是有很多人写require习惯了,而且不经常接触OOP模式,所以不习惯。 
其实使用use比require 好很多,还有很多人用require引入变量,这是大大错误,这是一种程序上编写的失误,所以建议以后大家不要用这种方式。编译的时候也不要用这种方式。那么用什么方式?如果你是一个有经验的Perl程序员,你应该知道。使用OPEN函数,传送变量值。这是编译Perl程序的关键,一些定量(不变的量),最好放在程序内部,变量以及客户所需要设置的量使用我先前说的那种方式。具体实践方法: 
Tanshuai OpenConf 函数代码: 
sub Open_Conf { 
open(FILE, "$_[0]");#打开~调用函数的文件名 
my @Conf_Info = ;#赋予~文件内容到@Conf_Info数组中 
close(FILE);#关闭~文件 
my $Conf_Infos ;定义~局部变量 

foreach $Conf_Infos (@Conf_Info) {#循环 
($name, $value) = split(/=/, $Conf_Infos);#区分~名称和数值 
($value, $dot) = split(/;/, $value);#区分~结束符 
$value=~s"'""gi;#删除~不必要的符号 
$CFG{$name} = $value;#复制~参数到散列变量 



配置文件原形: 
Port='81'; 
IP="127.0.0.1"; 
Listen='5'; 
调用方法: 
Open_Conf('../Conf/httpd.cfg');#../Conf/httpd.cfg为路径和文件名 
$port = $CFG{'Port'};#将文件原型的Port量复制到$port上,当然你可以不必这样做,可以直接引HASH 
$ip = $CFG{'IP'};#和上面的一样 

这样就解决了配置变量的问题,我想这个函数对某些人一定会有很重要的意义。 
在这里OOP就是use 方式的调用。 
现在我们要着重讨论OOP问题了,如果你不想把一大堆的程序代码写在一个文件中,那么使用OOP就最好了,原来是可以使用require,但这里不可一。OOP在Perl的好处显而易见,首先可以编译,即使不编译,它也同require有明显差异。 
OOP是在程序需要时调入,不需要时自动消失(通常说破坏对象)。require则不然,一旦调入一直存在,除非你使用exit 函数,所以在某些方面影响了程序的效率。 
例如我们要写一个Shell程序,一共需要一下部分:输入/输出(I/O)、命令判断(CMD)、System(系统操作)。 
我们平时也可以使用require,在编译的时候就好了,同样我们虽然可以按照子程序放在一个程序里面,但是在这里只是例子,但是在大宗商业项目中,这样做是显然费时费力的,会增加维护成本,无法联合开发等多种弊端。 
我们把他们分为4个文件3个模块一个主程序(编译):IO.pm、CMD.pm、System.pm、Shell.pl。 
首先要构造对象: 
Tanshuai 对象构造方法: 

package <包名或者对象名>; 
my %IN;#定义~包(对象)内部的散列 
sub new {#构造函数名 
my $class = shift; 
%IN= @_;#将调用对象的数值传入散列IN中 
my $self={}; 
bless $self,$class; 
return $self; 


虽然上面的构造有些不好的地方,但是它是通用对象的构造方法,利于调试,如果你认为没程序上的问题,就可以“封包”,适当修改变量传引方式。 
这里的所有对象只有是一个单一函数,只包括:构造对象和操作对象的两个部分,这是一个简单的对象引用,但是这种应用在实际的开发总是相当无畏的,在这里是为了方便教大家,所以不要什么程序都要对象。 

IO.pm: 

package IO; 
my %IN; 
sub new { 
my $class = shift; 
%IN = @_; 
my $self={}; 
bless $self,$class; 
return $self; 

sub do {#操作对象函数 
my $self=shift; 
defined ($_ = <>);#启动Shell得取输入信息 
chomp;#去掉无用的字符 
s/^\s+//;#过滤危险字符 
my $cmd = $_;#复制量 
return $cmd;#返回量 

1; 

CMD.pm: 

package CMD; 
my %IN; 
sub new { 
my $class = shift; 
%IN = @_; 
my $self={}; 
bless $self,$class; 
return $self; 

sub do {#操作对象函数 
my $self=shift; 
  my $cmd = @_ ;#传入调用程序的命令 
while (){#执行循环,直到退出 
if ($cmd eq 'ver') { 
print "Tanshuai Command Shell v.1.0.0.001225b\n"; 
print "(C)Copyright Tanshuai.Com 1997-2001\n"; 
print 'EMAIL:tanshuai@BIGFOOT.COM'; 
print "\n"; 
&do; 
}elsif ($cmd eq ""){ 
&do; 
exit; 
}elsif ($cmd eq 'exit'){ 
print "Exit System"; 
exit; 
}elsif ($cmd eq ‘dir'){ 
use System;#使用包System 
my $sys = System ::new ;#建立基于System包的对象$sys 
$sys->do($cmd) ;#操作对象sys传送命令 
&do; 
}else { 
print " Command Not Found "; 
&do;} 



1; 

System.pm: 

Package System; 
my %IN; 
sub new { 
my $class = shift; 
%IN = @_; 
my $self={}; 
bless $self,$class; 
return $self; 

sub do {#操作对象函数 
my $self=shift; 
my $cmd = @_ ; 
system($cmd) ;#使用System函数操作系统,启动dir命令 

1; 

以上各个模块(对象)已经建立完毕,我们现在只需要设计一个简单的操作对象程序。这个时候你发现搞对象原始是如此简单:) 

Shell.pl 主程序: 

use IO;#调用~模块(对象) IO.pm 
use CMD;#调用~模块(对象) CMD.pm 
my $IO = IO::new;#创建对象~$IO 
my $CMD = CMD::new;#创建对象~$CMD 
my $GetInput = $IO->do;#从对象IO得到输入信息; 
$CMD->do("$GetInput");#将得到的输入信息发送给对象$CMD,进行分析操作。 
exit ; 
  
这样就完成了,你可能问为什么没有使用对象System ?那是因为在对象CMD中继承对象System,所以我们不需要在程序中使用System,要不然就累了。 
当你看到shell.pl程序时候,你有何感想?是不是觉得搞对象简单了很多呢?给我的想法就是,以后程序员会越来越多,因为对象编程太简单了,而我们呢?哎,我们就去做对象。以后编程和做对象的人可能要区分开来了。 
现在编译shell.pl后,把这些对象删除,看看能否使用?当然能,假如你使用require就出现无法执行的致命错误。 
这里告诉大家编译Perl在较大或者较复杂的程序项目中,使用对象,会有很好的作用。你可能会问,用对象编译出来的程序如此之大,是否会影响效率?肯定会,但是它并非明显,就好像一个小小的15KB的程序,在运行的时候可能占用超过100MB的内存。由于它会整个被内存启动,但是并不会有较大幅度的效率下降。如果还想使用类似require的方法,就要看最后一章了。 



第六节 HTML模板编程方式——真正的WEB程序 
什么是真正的程序(Program)?我们平时使用ASP、PHP这些都不属于程序,它们只是一种页(Page),动态页面(Dynamic Page),但是我们一般称作页面编程(Web Programming),但这种说法不确切(并非不正确)。程序就是程序,并非所有的语言都叫做程序或编程语言。很多权威的书籍、文章和网站(例如:Yahoo!)都没有将ASP、PHP当作程序(编程语言)来解释。ASP是一种语言介质,PHP在Yahoo的定义页只是类似于SSI。他们说要做的东西顶多就是一个“后台(服务器端)的HTML(或者说是Script)”,可以想象,页(Page)和程序(Program)的差异,至少可以说页是由程序来解析输出结果的。那么也就是说,页想要做的事情比程序要局限得多。PHP不是一种程序,如果用ASP或PHP做一个Http服务器,你会有什么感觉?你见过吗?你见过ASP、PHP做的非Web“程序”吗?我想你没有见过。你相信用ASP、PHP编制出类似于Windows的图形(GUI)界面程序吗?那是一种什么感觉呢?所以,做程序和页面是两种不同的概念,在国内不知道是翻译的时候错误,还是大家都是这样理解的。 
如果你要写一个Web页,做一些小动作,用ASP、PHP、ePerl等未尝不可。但是它不是来给你做大宗Web项目或者软件而设计的。至少我是这样认为。而且我觉得Perl目前在程序中直接使用HTML是一种不好的习惯或者行为。它将增加维护成本,降低工作效率等诸多不便因素。其实我觉得外制式的模板方式的HTML套入法是适合时代潮流以及未来软件升级扩展的。至少可以让客户在不触及程序核心的前提下,随意修改界面,可以得到个性化、特性化的设置——未来趋势。而且我们可以降低很大维护的成本,同时某些不变的(诸如:版权、声明、标示)内容仍然可以使用内置式或者在套入模板的过程中进行相应修改等。如果你真的不喜欢他人修改模板,那么你可以使用加密方式,对模板文件进行加密,可以达到程序操作目的,和降低维护成本,而禁止他人修改的目的(推荐使用:Crypt::RC4)。 
本章将会着重讲述在Perl程序中(不但只是为了编译Perl)使用套入法,套入模板HTML,并且进行灵活的HTML操作。 
以下是标准的内置式和外制式的HTML操作: 
内置式HTML程序: 
#!perl 
$Var="HELLO WORLD"; 
print <Content-type: text/html 


$Var




HTML 
exit; 

外置式HTML程序: 
#!perl 
$Var="HELLO WORLD"; 
open (HTML,"../HelloWorld.html");#打开HelloWorld.html文件 
@HTML=; 
close (HTML); 
print Content-type :text/html ; 
foreach (@HTML) {#循环 
$_ =~ s/\*Var/$Var/g;#替换Hellworld.html 文件中*Var的内容为变量$Var的内容 
print "$_";#输出 

exit;   

外置式HTML文件 HellWorld.html: 


*Var





上面的例子都是现实操作中广泛(流行)用法,大家可能感觉到外置式有些复杂,其实不然,你只要把它做成一个函数或者对象就相当容易了。 
关键问题在于,变量的替换,若使用上面的方法,有些不妥,因为默写模板页面不一定是适合的那些变量,如果你把所有的变量都放在foreach里面,那么势必对于程序运行资源造成极大浪费,而且得不偿失,影响效率。这样做成一个函数或者对象,对会有不通用的问题。 
所以建立一个灵活的分析方法,对于模板HTML处理提供良好的快捷的运作模式。 
这个时候我们就要利用Perl强大的语法分析,来做一个自己的HTML语言分析语句了。这个语句看似简单缺令人头疼。 
我们现在以“*”符号作为模板中的变量(类似于Perl 中的$),这样有助于辨析。那么我想要把所有以“*”开头的变量,自动变换成程序内的对应变量,例如:要把*abc成为内部的$abc。一般情况我们需要逐个设置,这样大大浪费了时间,我们现在需要做一个通用的方法,无论什么的量都自动转换。这个语法很简单: 
$_ =~ /\*(\w+)/; 
看似简单的一局话,却有很大的作用,这句就是把以*开头的字符的名找出来,但是有趣的是,你不需要进行太复杂的,只要遇到空格或者其它非标准字符,就会自动排除。 
现在我们要把找到的字符名(即HTML的自定义变量)发给一个临时变量中(该步骤可以不做):$tmp = $1 ; 
现在要做的就是把这个*abc换成量$abc的值: 
$_ =~ s/\*$tmp/$Html{"$tmp"}/g 
这里的$Html是散列变量(HASH),为了方便和容易理解,我在这里采用HASH,这样对应的$Html{‘abc’}就被提出来,换掉*abc了。 
下面就是我做的模板套用函数与例子。 
打开文件的函数RTF: 
#!perl 
sub RTF{ 
open(READTXTFILE,"$_[0]"); 
@readtxtfile=
close(READTXTFILE); 
return @readtxtfile; 

  

分析模板的函数PHF: 
#!perl 
sub PHF { 
my $file = "$_[0]"; 
@HtmlFileMessages=&RTF("$file");#Open File; 
foreach (@HtmlFileMessages) { 
$_ =~ /\*(\w+)/; #替换网页的变量,批量处理,寻找“*”(*)标记 
$tmp = $1; #把寻找到的“*”标记以及其本身量名称复制给$tmp 
$_ =~ s/\*$tmp/$Html{"$tmp"}/g; #替换*$tmp内容成为%HTML哈希对应值 
print "$_"; 


分析模板的函数PHF: 
#!perl 
%Html (Var=> HelloWorld) ;#设置HTML文件的值 
PHF("../Helloworld.html ") ;#启动PHF函数 
这样就大功告成了,是不是比原来简单了很多呢?而且你可以更加容易的配置你的HTML 内的量与现实的量化分开来。这样你在编译Perl程序的时候可以节省很多事情。特别在用Perlcc 的时候。 



第七节 联合编译以及实例 
本章至关重要,你已经知道Perl的两种最好的编译方法。但是他们都有利弊,只要稍动脑筋,就可以实现“强强联合”,这样可以尽量避免那些缺憾。 
联合编译的道理很简单,但操作起来也不那样一帆风顺,其中有很多地方值得注意。联合编译主要有一个主程序和多个子程序(FILE)组成。它们之间是使用require函数连接。主程序只做连接等分析工作,子程序做细节工作,包括对象操作,模块引用。我们采用PerlCC 翻译C的方式来编译主程序成为一个可以执行的文件,在把子程序用Bytecode方式编译,这样即可免去无法使用部分模块的问题,也可以直接使用Perl程序,只要在主程序的前面定义一下模块引用路径,方法: 
use lib ‘<路径>’; 
这样就可以了,把那些需要调入的模块,放在制定路径中就好了。而且在CGI或者Socket的网络编程和页面编程中,使用该模是有助于提高效率,降低资源占用率。如果使用整体编译方法,那么每次启动必然会耗费相当大的内存,同样这个程序要重复关闭启动,做Fast CGI也是相当不方便的,这也是Fast CGI在Perl中的最好的方法。根据不同的请求套入不同的子程序。 
首先我们使用 cgi-lib.pl得去POST和GET数据(这个时候有些人会问,为什么不使用cgi.pm,我不是不想用它,而是cgi.pm在perlcc的任何编译模式都会有问题) 
然后根据不同的请求,我在这里设置为action。 
例如: 
require “cgi-lib”; 
if ($in{‘action’} eq “”) { 
require “display.pl”; 
&display; 
exit;#可选 
}elsif ($in{‘action’ }eq “love”) { 
require “love.pl”; 
&love; 
exit;#可选 

这样是很好的。我们使用perlcc 标准编译方法编译它,然后用-b模式编译display.pl和love.pl。然后把它们的名字改回.pl。 
注意在使用perlcc编译程序的时候,编译出来的程序必须带有应用程序扩展文件,如dll和so。因为你的程序还需它们支持,这个文件在Perl的解析软件目录下,例如perl5.6就是perl56.dll,必须把它拷贝到执行文件目录地下。在Linux下是.so。你最好在一个没有Perl 平台解析器的环境下进行测试,把那些需要使用的包也包括在里面。即使是VC等软件编译出来的程序,都需要在纯环境下测试,这是必要的。这样就可以测试出程序的一些不必要的问题。 
另外perlcc 的任何模式对语法都是很挑剔的,所以你最好使用比较正规的编写方法,而且单个perl程序如果程序量太大,必须截取到另一个文件中,否则编译后容易出现内存溢出现象。 
大家要知道如果你的子程序使用了ByteCode编译,但是他人仍然可以把你的子程序改成源代码形式,这样就好像我说的会被套出很多量。最好的的方法,是采用ByteCode 编译的程序写入一个Auth认证函数。当然最保险的方法是使用文件内容验证,但是效率影响,我认为不大必要。 
主程序: 
# !perl 
require “cgi-lib”; 
if ($in{‘action’} eq “”) { 
auth (“display.pl”); 
&display; 
exit;#可选 
}elsif ($in{‘action’ }eq “love”) { 
auth (“love.pl”); 
&love; 
exit;#可选 

sub auth { 
require "$_[0] " ; 
$auth = &check ; 
if ($auth ne "checkabcdefg "){ 
exit ; } 

Display.pl 
# !perl 
sub check { 
$check= "checkabcdefg " ; 
return $check ; 


sub display { 
print "content-type :text/html \n\n" ; 
print "hello baby " ; 


上面是一种简单的,不过也会造成一些问题,所以下面是一个麻烦(并非复杂)方法,但是很安全。 
检查编译程序是否真实: 
# !perl 
open (FILE,"./print.pl"); 
@FILE=
close (FILE); 
foreach (@FILE) { 
if ($_ =~/程序编译后的部分代码/){ 
}else {exit ;} 


首先把程序进行bytecode编译,然后截取部分独特的其它程序没有的代码,放入其中,来检查引入程序是否正确合法。 
你可以把bytecode的程序改名成.dll等,这样其它人就不知道是怎么回事啦。 
结束语 
Perl是一个强大的而且是最早的解析性程序语言,它的编译程序是B模块,大家可以详细常见,它有多种编译方式,都是采用反向编译(BackEnd)不同于反编译。所以经本上是不可能被反编译。我认为本文对所有的Perl程序员都有很大的帮助。 
Perl还有很多其它方式的编译、加密方法,但是我觉得本文介绍的几种方式都是最好的(兼容性和运行效率),有一些人,把写的程序进行部分字符乱码或者是取消缩近的书写格式(把所有程序写在一行上),我认为这些方法是“愚蠢的”,所以建议大家不要花那么多时间去研究这些“无谓”的东西。 
部分字符编码例子——原本: 
# !perl 
sub Hello { 
$hello=abc ; 
print $hello ; 

&hello ; 
部分字符编码例子——编码后 
#!perl 
sub adfjierei123489dkajd_dfefnkdj { 
$iernvmdnvcjnaldffgh=abc; 
print $iernvmdnvcjnaldffgh; 

&adfjierei123489dkajd_dfefnkdj; 

我希望通过本文促使Perl在国内的商业发展,也同样加快了Perl技术在国内的发展速度。但是我仍然希望大家可以写更多的公开源代码的程序出来,这样可以让初学者有较快的提高速度。      

▲评论

X 正在回复:
姓 名: 留下更多信息
性 别:
邮 件:
主 页:
Q Q:
来 自:
职 业:
评 论:
验 证:


Valid HTML 4.01 Strict Valid CSS!
Copyleft.A!die Software Studio.ADSS
Power by webmaster@adintr.com