常用Perl命令行參數應用介紹
本文和大家重點討論一下Perl命令行的概念,在學習Perl語言的過程中,Perl命令行問題經常會遇到,Perl有很多Perl命令行參數.通過它,我們有機會寫出更簡單的程序.在這篇文章里我們來了解一些常用的參數。
Perl命令行應用介紹
Perl有很多Perl命令行參數.通過它,我們有機會寫出更簡單的程序.在這篇文章里我們來了解一些常用的參數.
SafetyNetOptions
在使用Perl嘗試一些聰明(或stupid)的想法時,錯誤難免會發(fā)生.有經驗的Perl程序員常常使用三個參數來提前找到錯誤所在,
-C是第一個.這個參數編譯Perl程序但不會真正運行它.由此檢查所有語法錯誤.每次修改perl程序之后我都會立刻使用它來找到任何語法錯誤.
$perl-cprogram.pl
-W是第二個參數.它會提示你任何潛在的問題.Perl5.6.0之后的版本已經用usewarnings;替換了-w.你應該使用usewarnings因為它要比-w更靈活.
-T是第三個參數.它把perl放到了tain模式.在這個模式里,Perl會質疑任何程序外傳來的數據.例如,從Perl命令行讀取,外部文件里讀取或是CGI程序里傳來的數據.這些數據在-T模式里都會被Tainted掉.
Tainted數據不可以被用來和外部交互.例如使用在system調用和用作open的文件名.perlsec文檔里有更多什么數據會被Tainted掉的例子.
要想使用Tainted的數據就需要untaint這個數據.untaint是通過正則表達式來實現.這里我不會太多的講述taint模式.如果你要編寫的程序(例如CGI程序)需要從從用戶那里接受不可知的輸入,我推薦使有taint模式
-d,PerlDebugger,在這里值得一提但我們無法顧及,我推薦閱讀文檔'perldocperldebug'或RichardFoley的PerlDebuggerPocketReference一書.
Command-LinePrograms
下面的幾個Perl參數可以讓短小的Perl程序運行在Perl命令行.-e可以讓Perl程序在Perl命令行上運行.例如,我們可以在Perl命令行上運行"HelloWorld"程序而不用把它寫入文件再運行.
$perl-e'print"HelloWorld\n"'
多個-e也可以同時使用,運行順序根據它出現的位置.
$perl-e'print"Hello";'-e'print"World\n"'
象所有的Perl程序一樣,只有程序的最后一行不需要以;結尾.
雖然你也可以象通常一樣引用模塊,但-M讓它變得更容易.
$perl-MLWP::Simple-e'printhead"http://www.example.com"'
-M模塊名和use模塊名一樣.如果不想引入模塊的缺省值,你可以使用-m.-m模塊名和use模塊名()一樣.例如下面這個例子,因為head函數是缺省引入,而使用-m時就不會,結果是沒有輸出.
$perl-mLWP::Simple-e'printhead"http://www.example.com"'
-m和-M通過=來引入某個模塊的特別函數.
$perl-MCGI=:standard-e'printheader'
這里,CGI.pm的":standard"被引入,header函數因此可以使用.要引入多個參數可以通過使用引號和逗號.
$perl-MCGI='header,start_html'-e'printheader,start_html'
這里我們引入了header和start_html函數。#p#
ImplicitLoops
-n和-p增加了循環(huán)的功能,使你可以一行一行來處理文件.
$perl-n-e'somecode'file1
這與下面的程序一樣.
- LINE:
- while(<>;){
- #yourcodegoeshere
- }
<>;打開Perl命令行里的文件,一行行的讀取.每一行缺省保存在$_
$perl-n-e'print"$.-$_"'file
上面的這一行可以寫成
- LINE:
- while(<>;){
- print"$.-$_"
- }
輸出當前行數$.和當前行$_.
-p可以讓上面的程序變得更容易.-p會輸出$_就像這樣
- LINE:
- while(<>;){
- #yourcodegoeshere
- }continue{
- printordie"-pdestination:$!\n";
- }
continue在這里保證print在每次循環(huán)都會被調用.
使用-p,我們的打印行數程序可以改為
$perl-p-e'$_="$.-$_"'
注意到那個LINE:標簽?我們可以利用它來跳到下一個循環(huán).使用nextLINE
$perl-n-e'nextLINEunless/pattern/;print$_'
如果想在循環(huán)的前后做些處理,可以使用BEGIN或ENDblock.下面的這一行計算文件里的字數.
$perl-ne'END{print$t}@w=/(\w+)/g;$t+=@w'file.txt
每一行所有匹配的字放入數組@w,然后把@w的元素數目遞加到$t.ENDblock里的print最后輸出文件總字數.
還有兩個參數可以讓這個程序變得更簡單.-a打開自動分離(split)模式.空格是缺省的分離號.輸入根據分離號被分離然后放入缺省數組@F.由此,我們可以把上面的程序改寫為
$perl-ane'END{print$x}$x+=@F'file.txt
你也可以通過-F把缺省的分離號改為你想要的.例如把分離號定為非字符:
$perl-F'\W'-ane'END{print$x}$x+=@F'file.txt
下面通過Unixpassword文件來介紹一個復雜的例子.Unixpassword是文本文件,每一行是一個用戶記錄,由冒號:分離.第7行是用戶的登錄shell路徑.我們可以得出每一個不同shell路徑被多少個用戶使用:
- $perl-F':'-ane'$s{$F[6]}++;'\
- >;-e'END{print"$_:$s{$_}"forkeys%s}'/etc/passwd
雖然現在不是一行,但是你可以看出使用參數可以解決什么問題.
RecordSeparators
以前我提到過$/和$\--輸入,輸出分隔號.$/用來分隔從文件句柄里讀出的數據,缺省$/分隔號是\n,這樣每次從文件句柄里就會一行行的讀取.$\缺省是空字符,用來自動加到要print的數據尾端.這就是為什么很多時候print都要在末尾加上\n.
$/和$\可與-n-p一起使用.在Perl命令行上相對應為-0(零)和-l(這是L).-0后面可以跟一個16進制或8進制數值,這個值用來付給$/.-00打開段落模式,-0777打開slurp模式(即可以一次把整個文件讀入),這與把$/設為空字符和undef一樣效果.
單獨使用-l有兩個效果,第一自動chomp輸入分隔號,第二把$/值付給$\(這樣print的時候就會自動在末尾加\n)
我個人常常使用-l參數,用來給每一個輸出加\n.例如
- $perl-le'print"HelloWorld"'
- [size=18]In-PlaceEditing[/size]
使用已有的參數我們可以寫出很有效的Perl命令行程序.常見的UnixI/O重定向:
$perl-pe'somecode'<input.txt>;output.txt
這個程序從input.txt讀取數據,然后做一些處理再輸出到output.txt.你當然也可以把輸出重定向到同一個文件里.
上面的程序可以通過-i參數做的更簡單些.-i把源文件更名然后從這個更名的源文件里讀取.最后把處理后的數據寫入源文件.如果-i后跟有其他字符串,這個字符串與源文件名合成后來生成一個新的文件名.此文件會被用來儲存原始文件以免被-i參數覆蓋.
這個例子把所有php字符替換為perl:
$perl-i-pe's/\bPHP\b/Perl/g'file.txt
程序讀取文件的每一行,然后替換字符,處理后的數據重新寫入(即覆蓋)源文件.如果不想覆蓋源文件,可以使用
$perl-i.bak-pe's/\bPHP\b/Perl/g'file.txt
這里處理過的數據寫入file.txt,file.txt.bak是源文件的備份.
【編輯推薦】