📄 perl9.htm
字号:
<html><!-- #BeginTemplate "/Templates/index.dwt" --><head><meta http-equiv=Content-Type content="text/html; charset=gb_2312-80"><title>ChinaPerl</title>
<LINK rel=stylesheet href=mystyle.css type=text/css>
</head><body>
<table border=0 cellspacing=0 bgcolor=#80FFFF width=100%>
<tr>
<td align=left rowspan=2 width=150>
<div align="center"><img src=image/perl.gif height=85 width="85"></div>
</td>
<td align=center><img src=image/perl.gif height=85 width="85"></td>
</tr><tr>
<td colspan=2 align=center> </td>
</tr></table>
<table border=0 width=100% height=400 cellspacing=0 cellpadding=0>
<tr>
<td width=20% valign=top bgcolor="#dddddd">
<table border=0 width=100% cellpadding=0><tr><td width=5%></td>
<td width=95%>
<p align=center>Perl 5教程<br>
<a target=_blank href=http://flamephoenix.126.com>by flamephoenix</a><a href="http://perl.yesky.net"><br>
<font color="red">修订 by RNY<br>
</font></a>2000-08-30</p>
<p align=center><a href="index.htm">目录</a></p>
<p align=center>第一部分 Perl语言</p>
<p><a href=perl1.htm>第一章 概述</a></p>
<p><a href=perl2.htm>第二章 简单变量</a></p>
<p><a href=perl3.htm>第三章 操作符</a></p>
<p><a href=perl4.htm>第四章 列表和数组变量</a></p>
<p><a href=perl5.htm>第五章 文件读写</a></p>
<p><a href=perl6.htm>第六章 模式匹配</a></p>
<p><a href=perl7.htm>第七章 控制结构</a></p>
<p><a href=perl8.htm>第八章 子程序</a></p>
<p><a href=perl9.htm>第九章 关联数组(哈希表)</a></p>
<p><a href=perl10.htm>第十章 格式化输出</a></p>
<p><a href=perl11.htm>第十一章 文件系统</a></p>
<p><a href=perl12.htm>第十二章 引用(指针)</a></p>
<p><a href=perl13.htm>第十三章 面向对象编程</a></p>
<p><a href=perl14.htm>第十四章 包和模块</a></p>
<p><a href=func.htm>附录一 函数集</a></p>
<p align=center>第二部分 Perl的CGI应用</p>
<p><a href=cgi1.htm>第一章 cgilib例</a></p>
<p><a href=cgi2.htm>第二章 动态创建图像</a></p>
<p><a href=cgi3.htm>第三章 计数器的编写方法</a></p>
<p><a href=cgi4.htm>第四章 Web Server设置</a></p>
第五章、...<br></td></tr></table></td>
<td width=80% valign=top>
<div align="center"></div>
<table border=0 width=100% cellpadding=0>
<tr bgcolor="#00FF80">
<td colspan="2">
<div align="center"> Perl教程 | <a href="perl5manual.zip">下载全文</a>
| <a target=_blank href=http://www.turbolinux.com.cn/cgi-bin/linux/wwwthreads.pl?action=list&Board=perl>Perl论坛</a>
| <a href="http://perl.yesky.net">推荐站点</a> | 关于我们 </div>
</td>
</tr>
<tr>
<td colspan="2"><!-- #BeginEditable "body" -->
<p align=center>第九章 关联数组/哈希表</p>
<p align=center><a target=_blank href=http://flamephoenix.126.com>by
flamephoenix</a></p>
<a href=#1>一、数组变量的限制</a><br>
<a href=#2>二、定义</a><br>
<a href=#3>三、访问关联数组的元素</a><br>
<a href=#4>四、增加元素</a><br>
<a href=#5>五、创建关联数组</a><br>
<a href=#6>六、从数组变量复制到关联数组</a><br>
<a href=#7>七、元素的增删</a><br>
<a href=#8>八、列出数组的索引和值</a><br>
<a href=#9>九、用关联数组循环</a><br>
<a href=#10>十、用关联数组创建数据结构</a><br>
<a href=#10.1>1、(单)链表</a><br>
<a href=#10.2>2、结构</a><br>
<a href=#10.3>3、树</a><br>
<br>
<a name=1>一、数组变量的限制</a><br>
在前面讲的数组变量中,可以通过下标访问其中的元素。例如,下列语句访问数组@array的第三个元素:<br>
$scalar = $array[2];<br>
虽然数组很有用,但它们有一个显著缺陷,即很难记住哪个元素存贮的什么内容。假如我们来写一个程序计算某文件中首字母大写的单词出现的次数,用数组来实现就比较困难,程序代码如下:
<blockquote>1 : #!/usr/local/bin/perl<br>
2 :<br>
3 : while ($inputline = <STDIN>) {<br>
4 : while ($inputline =~ /\b[A-Z]\S+/g) {<br>
5 : $word = $&;<br>
6 : $word =~ s/[;.,:-]$//; # remove punctuation<br>
7 : for ($count = 1; $count <= @wordlist;<br>
8 : $count++) {<br>
9 : $found = 0;<br>
10: if ($wordlist[$count-1] eq $word) {<br>
11: $found = 1;<br>
12: $wordcount[$count-1] += 1;<br>
13: last;<br>
14: }<br>
15: }<br>
16: if ($found == 0) {<br>
17: $oldlength = @wordlist;<br>
18: $wordlist[$oldlength] = $word;<br>
19: $wordcount[$oldlength] = 1;<br>
20: }<br>
21: }<br>
22: }<br>
23: print ("Capitalized words and number of occurrences:\n");<br>
24: for ($count = 1; $count <= @wordlist; $count++) {<br>
25: print ("$wordlist[$count-1]: $wordcount[$count-1]\n");<br>
26: }</blockquote>
运行结果如下:<br>
<blockquote>Here is a line of Input.<br>
This Input contains some Capitalized words.<br>
^D<br>
Capitalized words and number of occurrences:<br>
Here: 1<br>
Input: 2<br>
This: 1<br>
Capitalized: 1</blockquote>
这个程序每次从标准输入文件读一行文字,第四行起的循环匹配每行中首字母大写的单词,每找到一个循环一次,赋给简单变量$word。在第六行中去掉标点后,查看该单词是否曾出现过,7~15行中在@wordlist中挨个元素做此检查,如果某个元素与$word相等,@wordcount中相应的元素就增加一个数。如果没有出现过,即@wordlist中没有元素与$word相等,16~20行给@wordlist和@wordcount增加一个新元素。<br>
<a name=2>二、定义</a><br>
正如你所看到的,使用数组元素产生了一些问题。首先,@wordlist中哪个元素对应着哪个单词并不明显;更糟的是,每读进一个新单词,程序必须检查整个列表才能知道该单词是否曾经出现过,当列表变得较大时,这是很耗费时间的。<br>
这些问题产生的原因是数组元素通过数字下标访问,为了解决这类问题,Perl定义了另一种数组,可以用任意简单变量值来访问其元素,这种数组叫做关联数组,也叫哈希表。<br>
为了区分关联数组变量与普通的数组变量,Perl使用%作为其首字符,而数组变量以@打头。与其它变量名一样,%后的第一个字符必须为字母,后续字符可以为字母、数字或下划线。<br>
<a name=3>三、访问关联数组的元素</a><br>
关联数组的下标可以为任何简单/标量值,访问单个元素时以$符号打头,下标用大括号围起来。例如:
<blockquote>$fruit{"bananas"}<br>
$number{3.14159}<br>
$integer{-7}</blockquote>
简单变量也可作为下标,如:<br>
$fruit{$my_fruit}<br>
<a name=4>四、增加元素</a><br>
创建一个关联数组元素最简单的方法是赋值,如语句$fruit{"bananas"} =
1; 把1赋给关联数组%fruit下标为bananas的元素,如果该元素不存在,则被创建,如果数组%fruit从未使用过,也被创建。<br>
这一特性使得关联数组很容易用于计数。下面我们用关联数组改写上面的程序,注意实现同样的功能此程序简化了许多。<br>
<blockquote>1 : #!/usr/local/bin/perl<br>
2 :<br>
3 : while ($inputline = <stdin>) {<br>
4 : while ($inputline =~ /\b[A-Z]\S+/g) {<br>
5 : $word = $&;<br>
6 : $word =~ s/[;.,:-]$//; # remove punctuation<br>
7 : $wordlist{$word} += 1;<br>
8 : }<br>
9 : }<br>
10: print ("Capitalized words and number of occurrences:\n");<br>
11: foreach $capword (keys(%wordlist)) {<br>
12: print ("$capword: $wordlist{$capword}\n");<br>
13: }</blockquote>
运行结果如下:<br>
<blockquote>Here is a line of Input.<br>
This Input contains some Capitalized words.<br>
^D<br>
Capitalized words and number of occurrences:<br>
This: 1<br>
Input: 2<br>
Here: 1<br>
Capitalized: 1</blockquote>
你可以看到,这次程序简单多了,读取输入并存贮各单词数目从20行减少到了7行。<br>
本程序用关联数组%wordlist跟踪首字母大写的单词,下标就用单词本身,元素值为该单词出现的次数。第11行使用了内嵌函数keys()。这个函数返回关联数组的下标列表,foreach语句就用此列表循环。<br>
注:关联数组总是随机存贮的,因此当你用keys()访问其所有元素时,不保证元素以任何顺序出现,特别值得一提的是,它们不会以被创建的顺序出现。<br>
要想控制关联数组元素出现的次序,可以用sort()函数对keys()返回值进行排列,如:
<blockquote>foreach $capword (sort keys(%wordlist)) {<br>
print ("$capword: $wordlist{$capword}\n");<br>
}</blockquote>
<a name=5>五、创建关联数组</a><br>
可以用单个赋值语句创建关联数组,如:<br>
%fruit = ("apples",17,"bananas",9,"oranges","none");<br>
此语句创建的关联数组含有下面三个元素:
<blockquote>
<li>下标为apples的元素,值为17</li>
<li>下标为bananas的元素,值为9</li>
<li>下标为oranges的元素,值为none</li>
</blockquote>
注:用列表给关联数组赋值时,Perl5允许使用"=>"或","来分隔下标与值,用"=>"可读性更好些,上面语句等效于:<br>
%fruit = ("apples"=>17,"bananas"=>9,"oranges"=>"none");<br>
<a name=6>六、从数组变量复制到关联数组</a><br>
与列表一样,也可以通过数组变量创建关联数组,当然,其元素数目应该为偶数,如:<br>
@fruit = ("apples",17,"bananas",9,"oranges","none");<br>
%fruit = @fruit;<br>
反之,可以把关联数组赋给数组变量,如:<br>
%fruit = ("grapes",11,"lemons",27);<br>
@fruit = %fruit;<br>
注意,此语句中元素次序未定义,那么数组变量@fruit可能为("grapes",11,"lemons",27)或("lemons",27,"grapes",11)。<br>
关联数组变量之间可以直接赋值,如:%fruit2 = %fruit1; 还可以把数组变量同时赋给一些简单变量和一个关联数组变量,如:<br>
($var1, $var2, %myarray) = @list;<br>
此语句把@list的第一个元素赋给$var1,第二个赋给$var2,其余的赋给%myarray。<br>
最后,关联数组可以通过返回值为列表的内嵌函数或用户定义的子程序来创建,下例中把split()函数的返回值--一个列表--赋给一个关联数组变量。
<blockquote>1: #!/usr/local/bin/perl<br>
2:<br>
3: $inputline = <STDIN>;<br>
4: $inputline =~ s/^\s+|\s+\n$//g;<br>
5: %fruit = split(/\s+/, $inputline);<br>
6: print ("Number of bananas: $fruit{\"bananas\"}\n");</blockquote>
运行结果如下:
<blockquote>oranges 5 apples 7 bananas 11 cherries 6<br>
Number of bananas: 11</blockquote>
<a name=7>七、元素的增删</a><br>
增加元素已经讲过,可以通过给一个未出现过的元素赋值来向关联数组中增加新元素,如$fruit{"lime"}
= 1;创建下标为lime、值为1的新元素。<br>
删除元素的方法是用内嵌函数delete,如欲删除上述元素,则:<br>
delete ($fruit{"lime"});<br>
注意:
<blockquote>1、一定要使用delete函数来删除关联数组的元素,这是唯一的方法。<br>
2、一定不要对关联数组使用内嵌函数push、pop、shift及splice,因为其元素位置是随机的。</blockquote>
<a name=8>八、列出数组的索引和值</a><br>
上面已经提到,keys()函数返回关联数组下标的列表,如:<br>
<blockquote>
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -