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解释器取其长度。
$scalar = @array; # $scalar = 3,即@array的长度
($scalar) = @array; # $scalar = 1,即@array第一个元素的值
注:以数组的长度为循环次数可做如下编程:
while ($count <= @array) {
print ("element $count: $array[$count-1]\n");
$count++;
}
6、子数组
@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
可以用子数组形式来交换元素:
7、有关数组的库函数
(1)sort--按字符顺序排序
@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(键盘)输入字符串时最后一个字符--换行符。而如果它作用到数组上,则将数组中每一个元素都做如此处理。
chop (@list); # @list = ("rabbi", "1234","quart") now
( 4)join/split--连接/拆分
join的第一个参数是连接所用的中间字符,其余则为待连接的字符数组。
@list = ("words","and");
$string = join("::", @list, "colons"); #结果为"words::and::colons"
@array = split(/::/,$string); # @array = ("words","and", "colons") now
-------------------------------------------------------
如果把标量认为是perl中的单数的话,那列表(list)和数组则可认为是perl中的复数。
列表是标量的有序集。数组是包含列表的变量。在perl 中这个两个术语是可以互换的。但严格意义上讲,列表是指数据,而数组是其变量名。可以有一些值(列表)但不属于数组;但每一个数组标量都有一个列表,虽然其可以为空。
列表中每一个元素都是一个独立的标量值。这些值是有顺序的,也就是说,这些值从开头到最后一个元素有一个固定的序列。数组或者列表中的元素是编了号的,其索引从整数0开始,依次增一,因此数组或者列表第一个元素的索引为0。
由于每一个元素是一个独立的标量值,因此一个列表或者数组可以包含数字,字符串,undef 值,或者任意不同类型的标量值的组合。然而,这些元素的类型通常是一致的。
列表和数组可以包含任意数量的元素。最少含有0元素,最多可以填满你的可用内存。这里又体现了perl 的设计哲学,“没有不必要的限制”。
1.访问数组元素
于perl 可以通过索引值来访问元素,如:$array[0] = “test”;print $array[0];
数组名字和标量是属于完全不同的命名空间(namespace)。同一程序也可以同时包含叫做$array的标量变量。perl将它们当作完全不同的事物来看待,不会混淆。(不建议这么做,容易弄混)
2.特殊的数组索引
如果将一个元素存储在数组最后元素的后面的位置,数组会自动增长的。perl 没有长度的限制,只要你有足够的内存。如果perl需要创建元素,则其值为undef。
有时需要知道数组最后一个元素的索引。如 array 数组,其最后一个元素的索引为$#array
一种简便方法:数组的负数索引值从最后一个元素开始。但不要认为这些索引是循环的。如果数组有3 元素,那有效的负数索引值是-1(最后一个元素),-2(中间的元素),-3(第一个元素)。实际上,几乎没有人使用除了-1之外的其它的负数索引值。
3.列表
数组是由括号括起来并且其元素由逗号分隔开的列表,这些值组成了数组的元素。
列表中的元素并非必须是常数,也可以是在执行此语句时再计算值的表达式。列表可以包含任意的标量值。
4.qw简写
实践表明,字符串的列表在perl中经常使用。有一种简便的方法可以不用输入大量的引号而达到类似的功能,那就是使用qw。
qw表示“quoted words”或者“quoted by whitespace,” 无论那种解释,perl 将它们当作单引号字符串处理,你不能像双引号那样在qw中使用\n 和$fred。whitespace(空格,像spaces,tabs,newlines 等字符串)将被忽略,剩下的组成了列表的元素。
由于qw是一种引用,因此不可以在qw内添加注释。
note:perl允许使用任何标点符号作为qw列表分界符。
5.列表赋值
和标量值类似,列表值也可以赋给变量:
($fred[0], $fred[1], $fred[2]) = (”haha”, “wawa”, undef);
print $fred[0].$fred[1].$fred[2];
左边列表中的每一个变量都得到了一个新值,和利用3 个赋值语句得到的结果是一样的。由于列表在赋值之前已经建立,因此在perl 中可以使用如下的简单方法交换两个变量的值。
note:当想引用这个数组时,perl 有一种简单的写法。在数组名前加@(后没有中括号)来引用整个数组。
6.pop和push操作
pop操作将数组的最后一个元素取出并返回:
@array = 5..9;
pop @array;
foreach $t (@array) {
print $t;
}
pop 后可以使用或者不使用括号。这在perl 中是一条通用规则:如果去掉括号含义不变,那括号就是可选的。
和pop 相反的操作是push,它可以将一个元素(或者一列元素)加在数组的末尾:
@array = 5..9;
push(@array,0);
foreach $t (@array) {
print $t;
}
note:push的第一个参数或者pop的唯一参数必须是数组变量。
7.shift和unshift操作
push和pop对数组的末尾进行操作,相应的,unshift和shift对一个数组的开头进行操作(数组的左端有最小下标的元素)。
8.将数组插入字符串
和标量类似,数组也可以插入双引号的字符串中。插入的数组元素会自动由空格分开。
@rocks = qw{wawa kaka haha};
print "@rocks";
9.perl 最常用的默认变量:$_
虽然它不是perl中唯一的默认变量,但无疑是使用的最普遍的。你将在许多例子中看到perl 在没有要求它使用某个变量或值时,会自动使用变量$_,这将节约程序员大量的时间来思考使用哪一个变量。
$_ ="yahoo google baidu";
print;
10.reverse操作
reverse(逆转)操作将输入的一串列表(可能是数组)按相反的顺序返回。
@fred = 6..10;
@barney = reverse(@fred);
print "@barney";
note:reverse返回逆转的列表,它不会改变其参数的值。如果返回值没有赋值给某个变量,那这个操作是没有什么意义的
11.sort操作
sort 操作将输入的一串列表(可能是数组)根据内部的字符顺序进行排序。
@rocks = qw/ bedrock slate rubble granite /;
@sorted = sort(@rocks);
print "@sorted";
12.标量和列表上下文
note:一个给定的表达式在不同的上下文(contexts)中其含义是不同的。
上下文是指表达式存在的地方。当perl解析表达式时,它通常期望一个标量值或者列表值。这既称为表达式的上下文环境。
@people = qw( fred barney betty );
@sorted = sort @people;
$number = 42 + @people;
print "@sorted";#列表context:barney , betty, fred
print "\n";
print "$number";#标量context:42+3,得到45
13.在标量context中使用list-producing表达式
@backwards = reverse qw / yabba dabba doo /;
print "@backwards"; #返回doo, dabba, yabba
print "\n";
$backwards = reverse qw/ yabba dabba doo /;
print "$backwards"; #返回oodabbadabbay
14.在列表context中使用scalar-producing表达式
如果一个表达式不是列表值,则标量值自动转换为一个元素的列表。
@fred = 6*7;
@barney = "hello".' '. "world";
print "@fred"; #返回42
print "@barney"; #返回hello world
15.强制转换为标量context
有时可能需要标量context 而perl期望的是列表。这种情况下,可以使用函数scalar。它不是一个真实的函数因为其仅是告诉perl提供一个标context:
@rocks = qw(talc quartz jade obsidian);
print "how many rocks do you have?\n";
print "i have ", @rocks, "rocks!\n"; #错误,输出rocks 的名字
print "i have ", scalar @rocks, "rocks!\n"; #正确,输出其数字
note:没有对应的函数能强制转换为列表context。
推荐阅读