Perl二維數(shù)組用法大全
本文和大家重點(diǎn)討論一下Perl二維數(shù)組的概念和用法,前面的例子里我們創(chuàng)建了包含有固定數(shù)據(jù)的Perl二維數(shù)組,但是如何往其中添加新元素呢?再或者如何從零開始創(chuàng)建一個(gè)Perl二維數(shù)組呢?本文回味大家一一講解。
Perl二維數(shù)組
最新版本可以從這里獲取(POD格式):
http://svn.Perlchina.org/trunk/POD2-CN/lib/POD2/CN/Perllol.pod
Perllol-操作數(shù)組的數(shù)組(Perl二維數(shù)組)
聲明和訪問數(shù)組的數(shù)組
創(chuàng)建一個(gè)數(shù)組的數(shù)組(有時(shí)也可以叫“列表的列表”,不過不太準(zhǔn)確)真是再簡單也不過了。它相當(dāng)容易理解,并且本文中出現(xiàn)的每個(gè)例子都有可能在實(shí)際應(yīng)用中出現(xiàn)。
數(shù)組的數(shù)組就是一個(gè)普通的數(shù)組(@AoA),不過可以接受兩個(gè)下標(biāo)("$AoA[3][2])。
下面先定義一個(gè)這樣的數(shù)組:"
#一個(gè)包含有“指向數(shù)組的引用”的數(shù)組
- @AoA=(
- ["fred","barney"],
- ["george","jane","elroy"],
- ["homer","marge","bart"],
- );
- print$AoA[2][2];
- bart
你可能已經(jīng)注意到,外面的括號是圓括號,這是因?yàn)槲覀兿胍o數(shù)組賦值,所以需要圓括號。如果你*不*希望這里是@AoA,而是一個(gè)指向它的引用,那么就得這樣:
#一個(gè)指向“包含有數(shù)組引用的數(shù)組”的引用
- $ref_to_AoA=[
- ["fred","barney","pebbles","bambam","dino",],
- ["homer","bart","marge","maggie",],
- ["george","jane","elroy","judy",],
- ];
- print$ref_to_AoA->[2][2];
注意外面的括號現(xiàn)在變成了方括號,并且我們的訪問語法也有所改變。這時(shí)因?yàn)楹虲不同,在Perl中你不能自由地交換數(shù)組和引用(在C中,數(shù)組和指針在很多地方可以互相代替使用)。$ref_to_AoA是一個(gè)數(shù)組引用,而@AoA是一個(gè)數(shù)組。同樣地,$AoA[2]也不是一個(gè)數(shù)組,而是一個(gè)數(shù)組引用。所以下面這兩行:
$AoA[2][2]
$ref_to_AoA->[2][2]
也可以用這兩行來代替:
$AoA[2]->[2]
$ref_to_AoA->[2]->[2]
這是因?yàn)檫@里有兩個(gè)相鄰的括號(不管是方括號還是花括號),所以你可以隨意地省略箭頭符號。但是如果$ref_to_AoA后面的那個(gè)箭頭不能省略,因?yàn)槭÷粤司蜎]法知道$ref_to_AoA到底是引用還是數(shù)組了^_^。#p#
修改Perl二維數(shù)組
前面的例子里我們創(chuàng)建了包含有固定數(shù)據(jù)的Perl二維數(shù)組,但是如何往其中添加新元素呢?再或者如何從零開始創(chuàng)建一個(gè)Perl二維數(shù)組呢?
首先,讓我們試著從一個(gè)文件中讀取Perl二維數(shù)組。首先我們演示如何一次性添加一行。首先我們假設(shè)有這樣一個(gè)文本文件:每一行代表了Perl二維數(shù)組的行,而每一個(gè)單詞代表了Perl二維數(shù)組的一個(gè)元素。下面的代碼可以把它們儲存到@AoA:
while(<>){
@tmp=split;
push@AoA,[@tmp];
}
你也可以用一個(gè)函數(shù)來一次讀取一行:
for$i(1..10){
$AoA[$i]=[somefunc($i)];
}
或者也可以用一個(gè)臨時(shí)變量來中轉(zhuǎn)一下,這樣看起來更清楚些:
for$i(1..10){
@tmp=somefunc($i);
$AoA[$i]=[@tmp];
}
注意方括號"[]"在這里非常重要。方括號實(shí)際上是數(shù)組引用的構(gòu)造器。如果不用方括號而直接寫,那就犯了很嚴(yán)重的錯誤:$AoA[$i]=@tmp;
你看,把一個(gè)數(shù)組賦值給了一個(gè)標(biāo)量,那么其結(jié)果只是計(jì)算了@tmp數(shù)組的元素個(gè)數(shù),我想這肯定不是你希望的。
如果你打開了"usestrict",那么你就得先定義一些變量然后才能避免警告:
- usestrict;
- my(@AoA,@tmp);
- while(<>){
- @tmp=split;
- push@AoA,[@tmp];
- }
當(dāng)然,你也可以不要臨時(shí)變量:
while(<>){
push@AoA,[split];
}
如果你知道想要放在什么地方的話,你也可以不要push(),而是直接進(jìn)行賦值:
- my(@AoA,$i,$line);
- for$i(0..10){
- $line=<>;
- $AoA[$i]=[split'',$line];
- }
甚至是這樣:
- my(@AoA,$i);
- for$i(0..10){
- $AoA[$i]=[split'',<>];
- }
你可能生怕<>在列表上下文會出差錯,所以想要明確地聲明要在標(biāo)量上下文中對<>求值,這樣可讀性會更好一些:(譯者注:列表上下文中,<>返回所有的行,標(biāo)量上下文中<>只返回一行。)
my(@AoA,$i);
for$i(0..10){
$AoA[$i]=[split'',scalar(<>)];
}
如果你想用$ref_to_AoA這樣的一個(gè)引用來代替數(shù)組,那你就得這么寫:
while(<>){
push@$ref_to_AoA,[split];
}
現(xiàn)在你已經(jīng)知道如何添加新行了。那么如何添加新列呢?如果你正在做數(shù)學(xué)中的矩陣運(yùn)算,那么要完成類似的任務(wù):
- for$x(1..10){
- for$y(1..10){
- $AoA[$x][$y]=func($x,$y);
- }
- }
- for$x(3,7,9){
- $AoA[$x][20]+=func2($x);
- }
想要訪問的某個(gè)元素是不是存在是無關(guān)緊要的:因?yàn)槿绻淮嬖谀敲碢erl會給你自動創(chuàng)建!新創(chuàng)建的元素的值是"undef"。
如果你想添加到一行的末尾,你可以這么做:
#添加新列到已存在的行
push@{$AoA[0]},"wilma","betty";
注意我*沒有*這么寫:
push$AoA[0],"wilma","betty";#錯誤!
事實(shí)上,上面這句根本就沒法通過編譯!為什么?因?yàn)閜ush()的第一個(gè)參數(shù)必須是一個(gè)真實(shí)的數(shù)組,不能是引用。#p#
訪問和打印
現(xiàn)在是打印Perl二維數(shù)組的時(shí)候了。那么怎么打???很簡單,如果你只想打印一個(gè)元素,那么就這么來一下:
print$AoA[0][0];
如果你想打印整個(gè)數(shù)組,那你可不能這樣:print@AoA;#錯誤!
因?yàn)槟氵@么做只能得到一列引用,Perl從來都不會自動地為你解引用。作為替代,你必須得弄個(gè)循環(huán)或者是雙重循環(huán)。用shell風(fēng)格的for()語句就可以打印整個(gè)Perl二維數(shù)組:
- for$aref(@AoA){
- print"\t[@$aref],\n";
- }
如果你要用下標(biāo)來遍歷的話,你得這么做:
- for$i(0..$#AoA){
- print"\telt$iis[@{$AoA[$i]}],\n";
- }
或者這樣用雙重循環(huán)(注意內(nèi)循環(huán)):
- for$i(0..$#AoA){
- for$j(0..$#{$AoA[$i]}){
- print"elt$i$jis$AoA[$i][$j]\n";
- }
- }
如同你看到的一樣,它有點(diǎn)兒復(fù)雜。這就是為什么有時(shí)候用臨時(shí)變量能夠看起來更簡單一些的原因:
- for$i(0..$#AoA){
- $aref=$AoA[$i];
- for$j(0..$#{$aref}){
- print"elt$i$jis$AoA[$i][$j]\n";
- }
- }
哦,好像還有點(diǎn)復(fù)雜,那么試試這樣:
- for$i(0..$#AoA){
- $aref=$AoA[$i];
- $n=@$aref-1;
- for$j(0..$n){
- print"elt$i$jis$AoA[$i][$j]\n";
- }
- }
切片
切片是指數(shù)組的一部分。如果你想要得到多維數(shù)組的一個(gè)切片,那你得進(jìn)行一些下標(biāo)運(yùn)算。通過箭頭可以方便地為單個(gè)元素解引用,但是訪問切片就沒有這么好的事了。當(dāng)然,我們可以通過循環(huán)來取切片。
我們先演示如何用循環(huán)來獲取切片。我們假設(shè)@AoA變量的值和前面一樣。
@part=();
$x=4;
for($y=7;$y<13;$y++){
push@part,$AoA[$x][$y];
}
這個(gè)循環(huán)其實(shí)可以用一個(gè)切片操作來代替:
@part=@{$AoA[4]}[7..12];
不過這個(gè)看上去似乎略微有些復(fù)雜。
下面再教你如何才能得到一個(gè)*二維切片*,比如$x從4到8,$y從7到12,應(yīng)該怎么寫?
- @newAoA=();
- for($startx=$x=4;$x<=8;$x++){
- for($starty=$y=7;$y<=12;$y++){
- $newAoA[$x-$startx][$y-$starty]=$AoA[$x][$y];
- }
- }
也可以省略掉中間的那層循環(huán):
for($x=4;$x<=8;$x++){
push@newAoA,[@{$AoA[$x]}[7..12]];
}
其實(shí)用map函數(shù)可以更加簡練:
@newAoA=map{[@{$AoA[$_]}[7..12]]}4..8;
雖然你的經(jīng)理也許會抱怨這種難以理解的代碼可能會帶來安全隱患,然而這種觀點(diǎn)還是頗有爭議的(興許還可以更加安全也說不定^_^)。
換了是我,我會把它們放進(jìn)一個(gè)函數(shù)中實(shí)現(xiàn):
- @newAoA=splice_2D(\@AoA,4=>8,7=>12);
- subsplice_2D{
- my$lrr=shift;#指向Perl二維數(shù)組的引用
- my($x_lo,$x_hi,
- $y_lo,$y_hi)=@_;
- returnmap{
- [@{$lrr->[$_]}[$y_lo..$y_hi]]
- }$x_lo..$x_hi;
- }