Usage of Array in Perl

一、列表
列表是包含在括号里的一序列的值,可以为任何数值,也可为空,如:(1, 5.3 , "hello" , 2),空列表:()。
注:只含有一个数值的列表(如:(43.2) )与该数值本身(即:43.2 )是不同的,但它们可以互相转化或赋值
列表例:
(17, $var, "a string")
(17, 26 << 2)
(17, $var1 + $var2)
($value, "The answer is $value")

二、数组--列表的存贮
列表存贮于数组变量中,与简单变量不同,数组变量以字符"@"打头,如:
@array = (1, 2, 3);
注:(1)数组变量创建时初始值为空列表:()。
(2)因为Perl用@和$来区分数组变量和简单变量,所以同一个名字可以同时用于数组变量和简单变量,如:
$var = 1;
@var = (11, 27.1 , "a string");

但这样很容易混淆,故不推荐
1、数组的存取
对数组中的值通过下标存取,第一个元素下标为0。试图访问不存在的数组元素,则结果为NULL,但如果给超出数组大小的元素赋值,则数组自动增长,原来没有的元素值为NULL。如:
@array = (1, 2, 3, 4);
$scalar = $array[0];
$array[3] = 5; # now @array is (1,2,3,5)
$scalar = $array[4]; # now $scalar = null;
$array[6] = 17; # now @array is (1,2,3,5,"","",17)

数组间拷贝
@result = @original;

用数组给列表赋值
@list1 = (2, 3, 4);
@list2 = (1, @list1, 5); # @list2 = (1, 2, 3, 4, 5)

数组对简单变量的赋值
(1) @array = (5, 7, 11);
($var1, $var2) = @array; # $var1 = 5, $var2 = 7, 11被忽略
(2) @array = (5, 7);
($var1, $var2, $var3) = @array; # $var1 = 5, $var2 = 7, $var3 ="" (null)
从标准输入(STDIN)给变量赋值
$var = ;
@array = ; # ^D为结束输入的符号

2、字符串中的方括号和变量替换
"$var[0]" 为数组@var的第一个元素。
"$var\[0]" 将字符"["转义,等价于"$var". "[0]",$var被变量替换,[0]保持不变。
"${var}[0]" 亦等价于"$var" ."[0]"。
"$\{var}"则取消了大括号的变量替换功能,包含文字:${var}.


3、列表范围:
(1..10) = (1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
(2, 5..7, 11) = (2, 5, 6, 7, 11)
(3..3) = (3)
用于实数
(2.1..5.3) = (2.1, 3.1 ,4.1, 5.1)
(4.5..1.6) = ()
用于字符串
("aaa".."aad") = ("aaa","aab", "aac", "aad")
@day_of_month = ("01".."31")
可包含变量或表达式
($var1..$var2+5)
.小技巧:
$fred = "Fred";
print (("Hello, " . $fred . "!\n") x 2);
其结果为:
Hello, Fred!
Hello, Fred!

4、数组的输出:
(1) @array = (1, 2, 3);
print (@array, "\n");
结果为:
123
(2) @array = (1, 2, 3);
print ("@array\n");
结果为:
1 2 3

5、列表/数组的长度
当数组变量出现在预期简单变量出现的地方(右值),则Perl解释器取其长度。
@array = (1, 2, 3);
$scalar = @array; # $scalar = 3,即@array的长度
($scalar) = @array; # $scalar = 1,即@array第一个元素的值

注:以数组的长度为循环次数可如下编程:
$count = 1;
while ($count <= @array) {
print ("element $count: $array[$count-1]\n");
$count++;
}

6、子数组
@array = (1, 2, 3, 4, 5);
@subarray = @array[0,1]; # @subarray = (1, 2)
@subarray2 = @array[1..3]; # @subarray2 = (2,3,4)
@array[0,1] = ("string", 46); # @array =("string",46,3,4,5) now
@array[0..3] = (11, 22, 33, 44); # @array = (11,22,33,44,5) now
@array[1,2,3] = @array[3,2,4]; # @array = (11,44,33,5,5) now
@array[0..2] = @array[3,4]; # @array = (5,5,"",5,5) now

可以用子数组形式来交换元素:
@array[1,2] = @array[2,1];

7、有关数组的库函数
(1)sort-- 按字符顺序排序
@array = ("this", "is", "a","test");
@array2 = sort(@array); # @array2 = ("a","is", "test", "this")
@array = (70, 100, 8);
@array = sort(@array); # @array = (100, 70, 8) now

(2)reverse--反转数组
@array2 = reverse(@array);
@array2 = reverse sort (@array);

(3)chop--数组去尾
chop的意义是去掉STDIN(键盘)输入字符串时最后一个字符--换行符。而如果它作用到数组上,则将数组中每一个元素都做如此处理。
@list = ("rabbit", "12345","quartz");
chop (@list); # @list = ("rabbi", "1234","quart") now

(4)join/split--连接/拆分
join的第一个参数是连接所用的中间字符,其余则为待连接的字符数组。
$string = join(" ", "this", "is","a", "string"); # 结果为"this is a string"
@list = ("words","and");
$string = join("::", @list, "colons"); #结果为"words::and::colons"
@array = split(/::/,$string); # @array = ("words","and", "colons") now

---------------------------------------------------------
push:从数组的末尾加入元素。
pop :从数组的末尾取出元素
shift:从数组的开头取出元素
unshift:从数组的开头加入元素

例子如下:
1),push
#!/usr/bin/Perl
#######
use strict;
use warnings;

my @array = ();
for ( my $i = 1 ; $i <= 5 ; ++$i ) {
push @array, $i;
print "@array\n";
}

[root@drbd-2 crontab]# ./push.pl
1
1 2
1 2 3
1 2 3 4
1 2 3 4 5

2),pop
#!/usr/bin/Perl
#######
use strict;
use warnings;

my @array = ( 1, 2, 3, 4, 5, 6 );
while (@array) {
my $firstTotal = pop(@array);
print "@array\n";
}

[root@drbd-2 crontab]# ./pop.pl
1 2 3 4 5
1 2 3 4
1 2 3
1 2
1

3),shift
#!/usr/bin/Perl
#######
use strict;
use warnings;

my @array = ( 1, 2, 3, 4, 5, 6 );
while (@array) {
my $firstTotal = shift(@array);
print "@array\n";
}

[root@drbd-2 crontab]# ./shift.pl
2 3 4 5 6
3 4 5 6
4 5 6
5 6
6

4),unshift
#!/usr/bin/Perl
#######
use strict;
use warnings;

my @array = ();
for ( my $i = 1; $i <= 5; ++$i ) {
unshift( @array, $i );         # add $i to front of @array
print "@array\n";              # display current @array
}

[root@drbd-2 crontab]# ./unshift.pl
1
2 1
3 2 1
4 3 2 1
5 4 3 2 1

5),splice
操作数组中间部分的函数,该函数主要有2个作用:

(1),向数组中间插入内容
#!/usr/bin/Perl
use strict;
use warnings;

my @array = ( 0 .. 6 );
my @array1 = ( 'a' .. 'd' );
my @replaced = splice( @array, 3, 2, @array1 );

print "replaced:     @replaced\n",
"with:         @array1\n",
"resulting in: @array\n\n";

[root@drbd-2 crontab]# ./splice.pl
replaced:     3 4
with:         a b c d
resulting in: 0 1 2 a b c d 5 6

(2),删除数组元素
#!/usr/bin/Perl
use strict;
use warnings;

my @array  = ( 0 .. 6 );
my @array1 = ( 'a' .. 'd' );
my @replaced = splice( @array, 3, 2 );

print "replaced:     @replaced\n",
"resulting in: @array\n\n";

[root@drbd-2 crontab]# ./splice.pl
replaced:     3 4
with:         a b c d
resulting in: 0 1 2 5 6

删除到末尾
#!/usr/bin/Perl
use strict;
use warnings;

my @array  = ( 0 .. 6 );
my @array1 = ( 'a' .. 'd' );
my @replaced = splice( @array, 3 );

print "replaced:     @replaced\n",
"resulting in: @array\n\n";

[root@drbd-2 crontab]# ./splice.pl
replaced:     3 4 5 6
resulting in: 0 1 2

6),join
连接列表中的各个分离的串,生成一个新的串,返回一个标量!

#!/usr/bin/Perl
use strict;
use warnings;

my @array = ( 0 .. 6 );
my $replaced = join("\n", @array);

print "$replaced\n",

[root@drbd-2 crontab]# ./join.pl
0
1
2
3
4
5
6

7),split
把字符串进行分割并把分割后的结果放入数组中
[root@drbd-2 crontab]# Perl -le '$p=q(/var/ftp/test);@a=split(/\/ftp\//,$p);print $a[1];'
test
[root@drbd-2 crontab]# Perl -le '$p=q(/var/ftp/test);@a=split(/\/ftp\//,$p);print $a[0];'
/var

8),scalar
统计数组的长度,一般我们不用这个,直接将数组赋值给标量即可。

#!/usr/bin/Perl
use strict;
use warnings;

my @array = ( 0 .. 6 );

my $count1 = @array;

my $count2 = scalar @array;


print "$count1\n";
print "$count2\n";

[root@drbd-2 crontab]# ./join.pl
7
7

9),sort
对数组元素进行排序

#!/usr/bin/Perl
use strict;
use warnings;

my @array    = ( 0 .. 9 );
my @reversed = reverse @array;
print "Original:    @array\n";
print "Reversed:    @reversed\n\n";

# create an unsorted array of numbers and sort it
my @array2            = ( 100, 23, 9, 75, 5, 10, 2, 50, 7, 96, 1, 40 );
my @sortedLexically   = sort @array2;
my @sortedNumerically = sort { $a $b } @array2;
print "Unsorted:    @array2\n";
print "Lexically:   @sortedLexically\n";
print "Numerically: @sortedNumerically\n";

[root@drbd-2 crontab]# ./join.pl
Original:    0 1 2 3 4 5 6 7 8 9
Reversed:    9 8 7 6 5 4 3 2 1 0
Unsorted:    100 23 9 75 5 10 2 50 7 96 1 40
Lexically:   1 10 100 2 23 40 5 50 7 75 9 96
# Numerically: 1 2 5 7 9 10 23 40 50 75 96 100

---------------------------------------------------------
Perl关联数组
关联数组和前面说的数组类似,它包含标量数据,可用索引值来单独选择这些数据,和数组不同的是,关联数组的索引值不是非负的整数而是任意的标量。这些标量称为Keys,可以在以后用于检索数组中的数值。
关联数组的元素没有特定的顺序,你可以把它们想象为一组卡片。每张卡片上半部分是索引而下半部分是数值。
关联数组是Perl 语言中特有的,关联数组是一个功能强大的数组。使用关联数组时要在前面加上 % 号,关联数组的格式如:
%ARRAY=(key1,value1,key2,value2,key3,value3);
每一个key 都有一个相对应的值(value)。
和数组类似,$zmd,@zmd,%zmd 之间没有任何联系。Perl 为对象的不同类型保留独立的命名空间。
下面介绍关联数组的操作:
在关联数组中增加、更改一对数据:ARRAY={key}=value; 在关联数组ARRAY 中加上一对key-value,要在关联数组名称前加上 $ 号,而且key 的名称要在 {} 符号之间,最后再指定key 所对应的 value 值。如果在这个关联数组中已经有这个key 了。就会更改这个key 所对应的 value。
keys(%array) 操作符可生成由关联数组%array中的所有当前关键字组成的列表。即返回奇数个元素(第1,3,5,7...个)的列表。
values(%array) 操作符返回由关联数组 %array 中所有当前值组成的列表,即返回偶数个列表。
echo(%array) 操作符返回由一个关键字和一个值对构成的两个元素的表。对同一数组再操作时返回下一对值直至结束。若没有更多的对时,echo() 返回空表。(这在打印全部列表中很有用)
删除一对在关联数组中的数据:delete $ARRAY{key};delete 是Perl 所提供的函数,作用是删除关联数组中的一个key以及这个key 所对应的 value。使用方法是 在delete 函数之后,指定关联数组中要删除的 key名称。

关联数组的综合举例:
%fred=(one,"zmd",two,"cxm"); $a=$fred{one}; --- #$a 等于 "zmd"
$b=$fred{two}---#$b 等于"cxm"
$fred{two}="yes"--- #%fred=(one,"zmd",two,"yes")

@number=keys(%fred)--- #@list=(one,two)
@user=values(%fred)---#@user=("zmd","cxm")
($number,$name)=echo(%fred)--- #此时 $number 的值为one,$name 的值为"zmd",再执行一次$number 为值为two,$name 的值为 "cxm")
@a=%fred--- #将关联数组fred 指定给数组a,这时 @a=(one,"zmd",two,"cxm")
%b=@a---#将数组a 指定给关联数组b,这时 %b=(one,"zmd",two,"cxm")
delete $fred{one}---#将关联数组中key 值为one 的一对key-value 删除,这时 %fred=(two,"cxm")

%name=()---#把 %name 关联数组置空


##########################################################################


Perl 中的二维数组:


声明和访问数组的数组


创建一个数组的数组(有时也可以叫“列表的列表”,不过不太准确)真是再简单也不过了。它相当容易理解,并且本文中出现的每个例子都有可能在实际应用中出现。


数组的数组就是一个普通的数组(@AoA),不过可以接受两个下标("$AoA[3][2])。
下面先定义一个这样的数组:"
#一个包含有“指向数组的引用”的数组
 


@AoA=(  
["fred","barney"],  
["george","jane","elroy"],  
["homer","marge","bart"],  
);  
 
print$AoA[2][2];  
bart 
你可能已经注意到,外面的括号是圆括号,这是因为我们想要给数组赋值,所以需要圆括号。如果你*不*希望这里是@AoA,而是一个指向它的引用,那么就得这样:


#一个指向“包含有数组引用的数组”的引用
 


$ref_to_AoA=[  
["fred","barney","pebbles","bambam","dino",],  
["homer","bart","marge","maggie",],  
["george","jane","elroy","judy",],  
];  
 
print$ref_to_AoA->[2][2];  
 
注意外面的括号现在变成了方括号,并且我们的访问语法也有所改变。这时因为和C不同,在Perl中你不能自由地交换数组和引用(在C中,数组和指针在很多地方可以互相代替使用)。$ref_to_AoA是一个数组引用,而@AoA是一个数组。同样地,$AoA[2]也不是一个数组,而是一个数组引用。所以下面这两行:


$AoA[2][2]
$ref_to_AoA->[2][2]


也可以用这两行来代替:


$AoA[2]->[2]
$ref_to_AoA->[2]->[2]



这是因为这里有两个相邻的括号(不管是方括号还是花括号),所以你可以随意地省略箭头符号。但是如果$ref_to_AoA后面的那个箭头不能省略,因为省略了就没法知道$ref_to_AoA到底是引用还是数组了^_^。




修改Perl二维数组


前面的例子里我们创建了包含有固定数据的Perl二维数组,但是如何往其中添加新元素呢?再或者如何从零开始创建一个Perl二维数组呢?


首先,让我们试着从一个文件中读取Perl二维数组。首先我们演示如何一次性添加一行。首先我们假设有这样一个文本文件:每一行代表了Perl二维数组的行,而每一个单词代表了Perl二维数组的一个元素。下面的代码可以把它们储存到@AoA:


while(<>){
@tmp=split;
push@AoA,[@tmp];
}


你也可以用一个函数来一次读取一行:


for$i(1..10){
$AoA[$i]=[somefunc($i)];
}


或者也可以用一个临时变量来中转一下,这样看起来更清楚些:


for$i(1..10){
@tmp=somefunc($i);
$AoA[$i]=[@tmp];
}


注意方括号"[]"在这里非常重要。方括号实际上是数组引用的构造器。如果不用方括号而直接写,那就犯了很严重的错误:$AoA[$i]=@tmp;


你看,把一个数组赋值给了一个标量,那么其结果只是计算了@tmp数组的元素个数,我想这肯定不是你希望的。


如果你打开了"usestrict",那么你就得先定义一些变量然后才能避免警告:


usestrict;  
my(@AoA,@tmp);  
while(<>){  
@tmp=split;  
push@AoA,[@tmp];  
}  
 
当然,你也可以不要临时变量:


while(<>){
push@AoA,[split];
}


如果你知道想要放在什么地方的话,你也可以不要push(),而是直接进行赋值:


my(@AoA,$i,$line);  
for$i(0..10){  
$line=<>;  
$AoA[$i]=[split'',$line];  
}  
 
甚至是这样:


my(@AoA,$i);  
for$i(0..10){  
$AoA[$i]=[split'',<>];  

你可能生怕<>在列表上下文会出差错,所以想要明确地声明要在标量上下文中对<>求值,这样可读性会更好一些:(译者注:列表上下文中,<>返回所有的行,标量上下文中<>只返回一行。)


my(@AoA,$i);
for$i(0..10){
$AoA[$i]=[split'',scalar(<>)];
}


如果你想用$ref_to_AoA这样的一个引用来代替数组,那你就得这么写:


while(<>){
push@$ref_to_AoA,[split];
}


现在你已经知道如何添加新行了。那么如何添加新列呢?如果你正在做数学中的矩阵运算,那么要完成类似的任务:


for$x(1..10){  
for$y(1..10){  
$AoA[$x][$y]=func($x,$y);  
}  
}  
 
for$x(3,7,9){  
$AoA[$x][20]+=func2($x);  
}  
 
想要访问的某个元素是不是存在是无关紧要的:因为如果不存在那么Perl会给你自动创建!新创建的元素的值是"undef"。


如果你想添加到一行的末尾,你可以这么做:


#添加新列到已存在的行
push@{$AoA[0]},"wilma","betty";


注意我*没有*这么写:


push$AoA[0],"wilma","betty";#错误!


事实上,上面这句根本就没法通过编译!为什么?因为push()的第一个参数必须是一个真实的数组,不能是引用。




访问和打印


现在是打印Perl二维数组的时候了。那么怎么打印?很简单,如果你只想打印一个元素,那么就这么来一下:


print$AoA[0][0];


如果你想打印整个数组,那你可不能这样:print@AoA;#错误!


因为你这么做只能得到一列引用,Perl从来都不会自动地为你解引用。作为替代,你必须得弄个循环或者是双重循环。用shell风格的for()语句就可以打印整个Perl二维数组:


for$aref(@AoA){  
print"\t[@$aref],\n";  

如果你要用下标来遍历的话,你得这么做:


for$i(0..$#AoA){  
print"\telt$iis[@{$AoA[$i]}],\n";  

或者这样用双重循环(注意内循环):


for$i(0..$#AoA){  
for$j(0..$#{$AoA[$i]}){  
print"elt$i$jis$AoA[$i][$j]\n";  
}  

如同你看到的一样,它有点儿复杂。这就是为什么有时候用临时变量能够看起来更简单一些的原因:


for$i(0..$#AoA){  
$aref=$AoA[$i];  
for$j(0..$#{$aref}){  
print"elt$i$jis$AoA[$i][$j]\n";  
}  
}  
 
哦,好像还有点复杂,那么试试这样:
for$i(0..$#AoA){  
$aref=$AoA[$i];  
$n=@$aref-1;  
for$j(0..$n){  
print"elt$i$jis$AoA[$i][$j]\n";  
}  

切片


切片是指数组的一部分。如果你想要得到多维数组的一个切片,那你得进行一些下标运算。通过箭头可以方便地为单个元素解引用,但是访问切片就没有这么好的事了。当然,我们可以通过循环来取切片。
我们先演示如何用循环来获取切片。我们假设@AoA变量的值和前面一样。


@part=();
$x=4;
for($y=7;$y<13;$y++){
push@part,$AoA[$x][$y];
}


这个循环其实可以用一个切片操作来代替:


@part=@{$AoA[4]}[7..12];


不过这个看上去似乎略微有些复杂。


下面再教你如何才能得到一个*二维切片*,比如$x从4到8,$y从7到12,应该怎么写?


 @newAoA=();  
for($startx=$x=4;$x<=8;$x++){  
for($starty=$y=7;$y<=12;$y++){  
$newAoA[$x-$startx][$y-$starty]=$AoA[$x][$y];  
}  
}  
 
也可以省略掉中间的那层循环:


for($x=4;$x<=8;$x++){
push@newAoA,[@{$AoA[$x]}[7..12]];
}


其实用map函数可以更加简练:


@newAoA=map{[@{$AoA[$_]}[7..12]]}4..8;


虽然你的经理也许会抱怨这种难以理解的代码可能会带来安全隐患,然而这种观点还是颇有争议的(兴许还可以更加安全也说不定^_^)。
换了是我,我会把它们放进一个函数中实现:


@newAoA=splice_2D(\@AoA,4=>8,7=>12);  
subsplice_2D{  
my$lrr=shift;#指向Perl二维数组的引用  
my($x_lo,$x_hi,  
$y_lo,$y_hi)=@_;  
 
returnmap{  
[@{$lrr->[$_]}[$y_lo..$y_hi]]  
}$x_lo..$x_hi;
 
}  

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值