五大常用Perl命令行參數(shù)應(yīng)用介紹
本文和大家重點(diǎn)討論一下Perl命令行應(yīng)用,主要包括五部分的內(nèi)容,Perl語言中有很多Perl命令行參數(shù).通過它們,我們有機(jī)會(huì)寫出更簡單的程序,在這篇文章里我們來了解一些常用的參數(shù)。
Perl命令行應(yīng)用介紹
Perl語言中有很多Perl命令行參數(shù).通過它們,我們有機(jī)會(huì)寫出更簡單的程序,在這篇文章里我們來了解一些常用的參數(shù)。
***部分:SafetyNetOptions安全網(wǎng)參數(shù)
在使用Perl嘗試一些聰明(或stupid)的想法時(shí),錯(cuò)誤難免會(huì)發(fā)生.有經(jīng)驗(yàn)的Perl程序員常常使用三個(gè)參數(shù)來提前找到錯(cuò)誤所在,
1:-C
這個(gè)參數(shù)編譯Perl程序但不會(huì)真正運(yùn)行它.由此檢查所有語法錯(cuò)誤.每次修改perl程序之后我都會(huì)立刻使用它來找到任何語法錯(cuò)誤.
$perl-cprogram.pl
2:-W
它會(huì)提示你任何潛在的問題.Perl5.6.0之后的版本已經(jīng)用usewarnings;替換了-w.你應(yīng)該使用usewarnings因?yàn)樗?w更靈活.
3:-T
它把perl放到了tain模式.在這個(gè)模式里,Perl會(huì)質(zhì)疑任何程序外傳來的數(shù)據(jù).例如,從Perl命令行讀取,外部文件里讀取或是CGI程序里傳來的數(shù)據(jù).
這些數(shù)據(jù)在-T模式里都會(huì)被Tainted掉.
第二部分:Perl命令行Perl參數(shù):可以讓短小的Perl程序運(yùn)行在Perl命令行.
1:-e
可以讓Perl程序在Perl命令行上運(yùn)行.
例如,我們可以在Perl命令行上運(yùn)行"HelloWorld"程序而不用把它寫入文件再運(yùn)行.
$perl-e'print"HelloWorld\n"'
多個(gè)-e也可以同時(shí)使用,運(yùn)行順序根據(jù)它出現(xiàn)的位置.
$perl-e'print"Hello";'-e'print"World\n"'
象所有的Perl程序一樣,只有程序的***一行不需要以;結(jié)尾.
2:-M
可以象通常一樣引用模
$perl-MLWP::Simple-e'getstore("http://www.163.com/","163.html")'##下載整個(gè)網(wǎng)頁
-M+模塊名和use模塊名一樣
第三部分:隱式循環(huán)
3:-n
增加了循環(huán)的功能,使你可以一行一行來處理文件
$perl-n-e'print;'1.txt#####$perl-ne'print;'1.txt
這與下面的程序一樣.
- LINE:
- while(<>;){
- print;
- }
<>;打開Perl命令行里的文件,一行行的讀取.每一行缺省保存在$_
$perl-n-e'print"$.-$_"'file
上面的這一行可以寫成
- LINE:
- while(<>;){
- print"$.-$_"
- }
輸出當(dāng)前行數(shù)$.和當(dāng)前行$_.
4:-p,和-n一樣,但是還會(huì)打印$_的內(nèi)容
如果想在循環(huán)的前后做些處理,可以使用BEGIN或ENDblock.下面的這一行計(jì)算文件里的字?jǐn)?shù).
$perl-ne'END{print$t}@w=/(\w+)/g;$t+=@w'file.txt
每一行所有匹配的字放入數(shù)組@w,然后把@w的元素?cái)?shù)目遞加到$t.ENDblock里的print***輸出文件總字?jǐn)?shù).
還有兩個(gè)參數(shù)可以讓這個(gè)程序變得更簡單.
5:-a
打開自動(dòng)分離(split)模式.空格是缺省的分離號(hào).輸入根據(jù)分離號(hào)被分離然后放入缺省數(shù)組@F
使用-a,上面的命令可以寫成這樣:
$perl-ane'END{print$x}$x+=@F'file.txt##使用了-a
6:-F
把缺省的分離號(hào)改為你想要的.例如把分離號(hào)定為非字符,上面的命令可以改為:
$perl-F'\W'-ane'END{print$x}$x+=@F'file.txt
下面通過Unixpassword文件來介紹一個(gè)復(fù)雜的例子.Unixpassword是文本文件,每一行是一個(gè)用戶記錄,
由冒號(hào):分離.第7行是用戶的登錄shell路徑.我們可以得出每一個(gè)不同shell路徑被多少個(gè)用戶使用:
$perl-F':'-ane'$s{$F[6]}++;'\>;-e'END{print"$_:$s{$_}"forkeys%s}'/etc/passwd
雖然現(xiàn)在不是一行,但是你可以看出使用參數(shù)可以解決什么問題.
第四部分:RecordSeparators數(shù)據(jù)分隔符
$/和$\--輸入,輸出分隔號(hào).
$/用來分隔從文件句柄里讀出的數(shù)據(jù),缺省$/分隔號(hào)是\n,這樣每次從文件句柄里就會(huì)一行行的讀取
$\缺省是空字符,用來自動(dòng)加到要print的數(shù)據(jù)尾端.這就是為什么很多時(shí)候print都要在末尾加上\n.
$/和$\可與-n-p一起使用.在Perl命令行上相對應(yīng)為-0(零)和-l(這是L).
-0后面可以跟一個(gè)16進(jìn)制或8進(jìn)制數(shù)值,這個(gè)值用來付給$/.
-00打開段落模式,-0777打開slurp模式(即可以一次把整個(gè)文件讀入),這與把$/設(shè)為空字符和undef一樣效果.
單獨(dú)使用-l有兩個(gè)效果:
***:自動(dòng)chomp輸入分隔號(hào)
第二:把$/值付給$\(這樣print的時(shí)候就會(huì)自動(dòng)在末尾加\n)
1:-l參數(shù),用來給每一個(gè)輸出加\n.例如
$perl-le'print"HelloWorld"'
第五部分:原位編輯
使用已有的參數(shù)我們可以寫出很有效的Perl命令行程序.常見的UnixI/O重定向:
$perl-pe'somecode'
這個(gè)程序從input.txt讀取數(shù)據(jù),然后做一些處理再輸出到output.txt.你當(dāng)然也可以把輸出重定向到同一個(gè)文件里.
上面的程序可以通過-i參數(shù)做的更簡單些.
2:-i
把源文件更名然后從這個(gè)更名的源文件里讀取.***把處理后的數(shù)據(jù)寫入源文件.
如果-i后跟有其他字符串,這個(gè)字符串與源文件名合成后來生成一個(gè)新的文件名.
此文件會(huì)被用來儲(chǔ)存原始文件以免被-i參數(shù)覆蓋.
這個(gè)例子把所有php字符替換為perl:
$perl-i-pe's/\bPHP\b/Perl/g'file.txt
程序讀取文件的每一行,然后替換字符,處理后的數(shù)據(jù)重新寫入(即覆蓋)源文件.
如果不想覆蓋源文件,可以使用
$perl-i.bak-pe's/\bPHP\b/Perl/g'file.txt
這里處理過的數(shù)據(jù)寫入file.txt,file.txt.bak是源文件的備份.
perl經(jīng)典的例子
問題:
遇到一問題:
aaa@domain.com2
aaa@domain.com111
bbb@home.com2222
bbb@home.com1
類似這種輸出,我想把他們變換成下面形式:
aaa@domain.com113
bbb@home.com2223
就是將相同郵箱名稱后面的數(shù)字相加。各位大俠能否給些思路如何用perl來實(shí)現(xiàn)。
答案:perl-anle'$cnt{$F[0]}+=$F[1];END{print"$_\t$cnt{$_}"forkeys%cnt}'urfile
如果熟悉了上面幾個(gè)Perl命令行參數(shù)的用法,上面的這個(gè)命令應(yīng)該很好理解:
每次讀取urfile的一行,由于使用了-a,打開自動(dòng)分離(split)模式.空格是缺省的分離號(hào).輸入根據(jù)分離號(hào)被分離然后放入缺省數(shù)組@F中,
以文件的***行為例子$F[0]就是aaa@domain.com,$F[1]就是2
$cnt{$F[0]}+=$F[1]就是一個(gè)哈希數(shù)組,以$F[0]為key,$F[1]為value,把相同key的數(shù)值都疊加起來.然后把文件的每一行都這樣處理一次.
END{}就是在循環(huán)完之后再處理.里面的意思就是打印這個(gè)%cnt哈希數(shù)組.這個(gè)哈希數(shù)組的key就是郵箱名稱,value就是疊加后的數(shù)字.
下面的是上面行命令的文本形式:
- #!/usr/bin/perl
- usestrict;
- usewarnings;
- my%hash;
- while(<>){
- chomp;
- my@array=split;
- $hash{$array[0]}+=$array[1];
- }
- END{
- foreach(keys%hash){
- print"$_\t$hash{$_}\n";
- }
- }
【編輯推薦】