Perl二維數(shù)組用法全程剖析
本文和大家重點(diǎn)討論一下PerlPerl二維數(shù)組的概念,PerlPerl二維數(shù)組簡(jiǎn)單說(shuō)就是數(shù)組的數(shù)組,創(chuàng)建一個(gè)數(shù)組的數(shù)組(有時(shí)也可以叫“列表的列表”,不過(guò)不太準(zhǔn)確)真是再簡(jiǎn)單也不過(guò)了。請(qǐng)看下面詳細(xì)介紹。
Perl二維數(shù)組
非常簡(jiǎn)短的一個(gè)Perl二維數(shù)組教程,由鄙人翻譯完成。
***版本可以從這里獲取(POD格式):
http://svn.perlchina.org/trunk/POD2-CN/lib/POD2/CN/perllol.pod
NAME
perllol-操作數(shù)組的數(shù)組(Perl二維數(shù)組)
說(shuō)明
Perl二維數(shù)組中聲明和訪問(wèn)數(shù)組的數(shù)組
創(chuàng)建一個(gè)數(shù)組的數(shù)組(有時(shí)也可以叫“列表的列表”,不過(guò)不太準(zhǔn)確)真是再簡(jiǎn)單也不過(guò)了。它相當(dāng)容易理解,并且本文中出現(xiàn)的每個(gè)例子都有可能在實(shí)際應(yīng)用中出現(xiàn)。
數(shù)組的數(shù)組就是一個(gè)普通的數(shù)組(@AoA),不過(guò)可以接受兩個(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)注意到,外面的括號(hào)是圓括號(hào),這是因?yàn)槲覀兿胍o數(shù)組賦值,所以需要圓括號(hào)。如果你*不*希望這里是@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];
注意外面的括號(hào)現(xiàn)在變成了方括號(hào),并且我們的訪問(wèn)語(yǔ)法也有所改變。這時(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]
也可以用這兩行來(lái)代替:
$AoA[2]->[2]
$ref_to_AoA->[2]->[2]
這是因?yàn)檫@里有兩個(gè)相鄰的括號(hào)(不管是方括號(hào)還是花括號(hào)),所以你可以隨意地省略箭頭符號(hào)。但是如果$ref_to_AoA后面的那個(gè)箭頭不能省略,因?yàn)槭÷粤司蜎](méi)法知道$ref_to_AoA到底是引用還是數(shù)組了^_^。#p#
修改Perl二維數(shù)組
前面的例子里我們創(chuàng)建了包含有固定數(shù)據(jù)的Perl二維數(shù)組,但是如何往其中添加新元素呢?再或者如何從零開始創(chuàng)建一個(gè)Perl二維數(shù)組呢?
首先,讓我們?cè)囍鴱囊粋€(gè)文件中讀取Perl二維數(shù)組。首先我們演示如何一次性添加一行。首先我們假設(shè)有這樣一個(gè)文本文件:每一行代表了Perl二維數(shù)組的行,而每一個(gè)單詞代表了Perl二維數(shù)組的一個(gè)元素。下面的代碼可以把它們儲(chǔ)存到@AoA:
while(<>){
@tmp=split;
push@AoA,[@tmp }
你也可以用一個(gè)函數(shù)來(lái)一次讀取一行:
for$i(1..10){
$AoA[$i]=[somefunc($i)];
}
或者也可以用一個(gè)臨時(shí)變量來(lái)中轉(zhuǎn)一下,這樣看起來(lái)更清楚些:
for$i(1..10){
@tmp=somefunc($i);
$AoA[$i]=[@tmp];
}
注意方括號(hào)"[]"在這里非常重要。方括號(hào)實(shí)際上是數(shù)組引用的構(gòu)造器。如果不用方括號(hào)而直接寫,那就犯了很嚴(yán)重的錯(cuò)誤:$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'',<>];
- }
你可能生怕<>在列表上下文會(huì)出差錯(cuò),所以想要明確地聲明要在標(biāo)量上下文中對(duì)<>求值,這樣可讀性會(huì)更好一些:(譯者注:列表上下文中,<>返回所有的行,標(biāo)量上下文中<>只返回一行。)
- my(@AoA,$i);
- for$i(0..10){
- $AoA[$i]=[split'',<>];
- }
如果你想用$ref_to_AoA這樣的一個(gè)引用來(lái)代替數(shù)組,那你就得這么寫:
while(<>){
push@$ref_to_AoA,[split];
}
現(xiàn)在你已經(jīng)知道如何添加新行了。那么如何添加新列呢?Perl二維數(shù)組中如果你正在做數(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);
- }
想要訪問(wèn)的某個(gè)元素是不是存在是無(wú)關(guān)緊要的:因?yàn)槿绻淮嬖谀敲碢erl會(huì)給你自動(dòng)創(chuàng)建!新創(chuàng)建的元素的值是"undef"。
如果你想添加到一行的末尾,你可以這么做:
#添加新列到已存在的行
push@{$AoA[0]},"wilma","betty";
注意我*沒(méi)有*這么寫:
push$AoA[0],"wilma","betty";#錯(cuò)誤!
事實(shí)上,上面這句根本就沒(méi)法通過(guò)編譯!為什么?因?yàn)閜ush()的***個(gè)參數(shù)必須是一個(gè)真實(shí)的數(shù)組,不能是引用。#p#
訪問(wèn)和打印
現(xiàn)在是打印Perl二維數(shù)組的時(shí)候了。那么怎么打印?很簡(jiǎn)單,如果你只想打印一個(gè)元素,那么就這么來(lái)一下:
print$AoA[0][0];
如果你想打印整個(gè)數(shù)組,那你可不能這樣:print@AoA;#錯(cuò)誤!
因?yàn)槟氵@么做只能得到一列引用,Perl從來(lái)都不會(huì)自動(dòng)地為你解引用。作為替代,你必須得弄個(gè)循環(huán)或者是雙重循環(huán)。用shell風(fēng)格的for()語(yǔ)句就可以打印整個(gè)Perl二維數(shù)組:
for$aref(@AoA){
print"\t[@$aref],\n";
}
如果你要用下標(biāo)來(lái)遍歷的話,你得這么做:
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í)變量能夠看起來(lái)更簡(jiǎn)單一些的原因:
- 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";
- }
- }
切片
切片是指Perl二維數(shù)組的一部分。如果你想要得到多維數(shù)組的一個(gè)切片,那你得進(jìn)行一些下標(biāo)運(yùn)算。通過(guò)箭頭可以方便地為單個(gè)元素解引用,但是訪問(wèn)切片就沒(méi)有這么好的事了。當(dāng)然,我們可以通過(guò)循環(huán)來(lái)取切片。
我們先演示如何用循環(huán)來(lái)獲取切片。我們假設(shè)@AoA變量的值和前面一樣。
@part=();
$x=4;
for($y=7;$y<13;$y++){
push@part,$AoA[$x][$y];
}
這個(gè)循環(huán)其實(shí)可以用一個(gè)切片操作來(lái)代替:
@part=@{$AoA[4]}[7..12];
不過(guò)這個(gè)看上去似乎略微有些復(fù)雜。
下面再教你Perl二維數(shù)組中如何才能得到一個(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ù)可以更加簡(jiǎn)練:
@newAoA=map{[@{$AoA[$_]}[7..12]]}4..8;
雖然你的經(jīng)理也許會(huì)抱怨這種難以理解的代碼可能會(huì)帶來(lái)安全隱患,然而這種觀點(diǎn)還是頗有爭(zhēng)議的(興許還可以更加安全也說(shuō)不定^_^)。換了是我,我會(huì)把它們放進(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;
}
【編輯推薦】
- Perl變量中Perl數(shù)組概念詳解
- 深入解析Perl中Perl數(shù)組的使用
- Perl學(xué)習(xí)筆記 Perl雙引號(hào)和單引號(hào)的區(qū)別
- 揭秘Perl變量中Perl純變量用法
- 深入解析Perl內(nèi)部函數(shù)的使用

















