.\" Automatically generated by Pod::Man v1.37, Pod::Parser v1.14 .\" .\" Standard preamble: .\" ======================================================================== .\".de Sh\" Subsection heading .\".br .\".if t .Sp .\".ne 5 .\".PP .\"\fB\\$1\fR .\".PP .\".. .\".de Sp\" Vertical space (when we can't use .PP) .\".if t .sp .5v .\".if n .sp .\".. .\".de Vb\" Begin verbatim text .\".ft CW .\".nf .\".ne\\$1 .\".. .\".de Ve\" End verbatim text .\".ft R .\".fi .\".. .\".\" Set up some character translations and predefined strings. \*(-- will .\".\" give an unbreakable dash,\*(PI will give pi,\[lq] will give a left .\".\" double quote, and\[rq] will give a right double quote. | will give a .\".\" real vertical bar. \*(C+ will give a nicer C++. Capital omega is used to .\".\" do unbreakable dashes and therefore won't be available. ` and' .\".\" expand to `' in nroff, nothing in troff, for use with C<>. .\".tr \(*W-|\(bv\*(Tr .\".ds C+ C\v'-.1v'\h'-1p'\s-2+\h'-1p'+\s0\v'.1v'\h'-1p' .\".ie n\{\ .\". ds --\(*W- .\". ds PI pi .\". if (\n(.H=4u)&(1m=24u) .ds --\(*W\h'-12u'\(*W\h'-12u'-\" diablo 10 pitch .\". if (\n(.H=4u)&(1m=20u) .ds --\(*W\h'-12u'\(*W\h'-8u'-\" diablo 12 pitch .\". ds L" "" .\". ds R" "" .\". ds C` "" .\". ds C' "" .\"'br\} .\".el\{\ .\". ds --\|\(em\| .\". ds PI\(*p .\". ds L" `` .\". ds R" '' .\"'br\} .\".\" .\".\" If the F register is turned on, we'll generate index entries on stderr for .\".\" titles (.TH), headers (.SH), subsections (.Sh), items (.Ip), and index .\".\" entries marked with X<> in POD. Of course, you'll have to process the .\".\" output yourself in some meaningful fashion. .\".if\nF\{\ .\". de IX .\". tm Index:\\$1\t\\n%\t"\\$2" .\".. .\". nr % 0 .\". rr F .\".\} .\".\" .\".\" For nroff, turn off justification. Always turn off hyphenation; it makes .\".\" way too many mistakes in technical documents. .\".hy 0 .\".if n .na .\".\" .\".\" Accent mark definitions (@(#)ms.acc 1.5 88/02/08 SMI; from UCB 4.2). .\".\" Fear. Run. Save yourself. No user-serviceable parts. .\". \" fudge factors for nroff and troff .\".if n\{\ .\". ds #H 0 .\". ds #V .8m .\". ds #F .3m .\". ds #[\f1 .\". ds #]\fP .\".\} .\".if t\{\ .\". ds #H ((1u-(\\\\n(.fu%2u))*.13m) .\". ds #V .6m .\". ds #F 0 .\". ds #[\& .\". ds #]\& .\".\} .\". \" simple accents for nroff and troff .\".if n\{\ .\". ds '\& .\". ds `\& .\". ds ^\& .\". ds ,\& .\". ds ~ ~ .\". ds / .\".\} .\".if t\{\ .\". ds '\\k:\h'-(\\n(.wu*8/10-\*(#H)'\'\h"|\\n:u" .\". ds `\\k:\h'-(\\n(.wu*8/10-\*(#H)'\`\h'|\\n:u' .\". ds ^\\k:\h'-(\\n(.wu*10/11-\*(#H)'^\h'|\\n:u' .\". ds ,\\k:\h'-(\\n(.wu*8/10)',\h'|\\n:u' .\". ds ~\\k:\h'-(\\n(.wu-\*(#H-.1m)'~\h'|\\n:u' .\". ds /\\k:\h'-(\\n(.wu*8/10-\*(#H)'\z\(sl\h'|\\n:u' .\".\} .\". \" troff and (daisy-wheel) nroff accents .\".ds :\\k:\h'-(\\n(.wu*8/10-\*(#H+.1m+\*(#F)'\v'-\*(#V'\z.\h'.2m+\*(#F'.\h'|\\n:u'\v'\*(#V' .\".ds 8\h'\*(#H'\(*b\h'-\*(#H' .\".ds o\\k:\h'-(\\n(.wu+\w'\(de'u-\*(#H)/2u'\v'-.3n'\*(#[\z\(de\v'.3n'\h'|\\n:u'\*(#] .\".ds d-\h'\*(#H'\(pd\h'-\w'~'u'\v'-.25m'\f2\(hy\fP\v'.25m'\h'-\*(#H' .\".ds D- D\\k:\h'-\w'D'u'\v'-.11m'\z\(hy\v'.11m'\h'|\\n:u' .\".ds th\*(#[\v'.3m'\s+1I\s-1\v'-.3m'\h'-(\w'I'u*2/3)'\s-1o\s+1\*(#] .\".ds Th\*(#[\s+2I\s-2\h'-\w'I'u*3/5'\v'-.3m'o\v'.3m'\*(#] .\".ds ae a\h'-(\w'a'u*4/10)'e .\".ds Ae A\h'-(\w'A'u*4/10)'E .\". \" corrections for vroff .\".if v .ds ~\\k:\h'-(\\n(.wu*9/10-\*(#H)'\s-2\u~\d\s+2\h'|\\n:u' .\".if v .ds ^\\k:\h'-(\\n(.wu*10/11-\*(#H)'\v'-.4m'^\v'.4m'\h'|\\n:u' .\". \" for low resolution devices (crt and lpr) .\".if\n(.H>23 .if\n(.V>19\ .\"\{\ .\". ds : e .\". ds 8 ss .\". ds o a .\". ds d- d\h'-1'\(ga .\". ds D- D\h'-1'\(hy .\". ds th\o'bp' .\". ds Th\o'LP' .\". ds ae ae .\". ds Ae AE .\".\} .\".rm #[ #] #H #V #F C .\".\" ======================================================================== .\".\" .\".IX Title "PERLDATA 1" .TH PERLDATA 7 "2003-11-25" "perl v5.8.3" "Perl Programmers Reference Guide" .SH "NAME" perldata\- Perl 數據類型 .SH "DESCRIPTION 描述" .\".IX Header "DESCRIPTION" .SS "Variable names 變量名" .\".IX Subsection "Variable names" Perl 有三種內建的數據類型:標量,數組和關聯數組(即\[lq]哈希表, hash\[rq])。數組以數字爲索引,通常以0開始,升序排列。哈希表以與值相關聯 的字符串爲索引,內部排列是無序的。 .PP 值通常通過一個變量名(或變量名的引用)來引用。變量名的前綴字符顯示了值的 數據類型。其餘部分指明瞭引用的是哪一個特定的值。通常變量名是一個唯一的標 識符,以字母或下劃線開始,包括字母、下劃線和數字。某些情況下,也可以是以 `::'\fR 分隔的一串標識符(或者是過時的`''\fR);除了最後一 個,其它都是包名,用來定位最後一個標識符所在的位置(詳情參見 perlmod 中的 Packages)。可以用一個簡單的標識符來替代它,利用引用就可以。下文有詳述, 也可參見 perlref . .PP Perl 也有內建的變量,其名稱不遵循這一規則。它們名稱古怪,這樣可以避免與你的變量名衝突。模式匹配中被匹配到的字符串是以`$'\fR 加一個數字的變量名來存放的(參見 the perlop manpage 和 the perlre manpage)。 另外,還有幾個使你可以介入perl 內部工作的特殊變量,其名稱中包含標點和控制字符(參見 perlvar ) .PP 標量以 '$'開始, 即使它是數組或哈希的元素也是如此。可以把 '$' 理解爲`s' ,表示scalar(標量)。(譯者注:此處根據有關文檔,做了改動,下面的@處也 是這樣) .PP .nf \& $days # 簡單標量 "days" \& $days[28] # 數組 @days的第29個元素 \& $days{'Feb'} # 哈希 %days的 `Feb' 所對應的值 \& $#days # 數組 @days的最後一個元素的索引值 .fi .PP 整個數組(或數組和哈希的局部)以 '@'開始, 它類似英文中的\[lq]these\[rq] 或\[lq]those\[rq] (這些...那些...),表示期望有多個值。 .PP .nf \& @days # ($days[0], $days[1],... $days[n]) \& @days[3,4,5] # 即 ($days[3],$days[4],$days[5]) \& @days{'a','c'} # 即 ($days{'a'},$days{'c'}) .fi .PP 整個哈希以 '%' 開始: .PP .nf \& %days # (key1, val1, key2, val2 ...) .fi .PP 另外,子程序以'&'來表示, 但有時在不引起誤解的情況下也可以不用, 就象 \[lq]do\[rq] 在英語中常常省略一樣。 符號表項以 '*' 作爲開始字符, 不過你 現在還不用關心這個 (if ever ;-) .PP 每一種數據類型都有它自己的名字空間,常量標識符也一樣。這意味着你可以使用 同一個名字來命名標量、數組、哈希、文件句柄、目錄句柄、子程序、格式或標籤。 即$foo\fR 和@foo\fR 是不同的變量。也即意味着$foo[1]\fR 是 @foo\fR 的一部分, 而不是$foo\fR的一部分. 這看來有些怪異,不過很 正常,因爲它本來就怪異。 .PP 因爲變量名以 '$', '@', 或 '%'開始, 保留詞對變量沒有什麼影響。保留詞影響 的是標籤和文件句柄,因爲它們不是以特殊字符前綴開始的。你不能用\[lq]log\[rq] 來命名文件句柄,因爲它是保留詞(提示:你可以用 `open(LOG,'logfile')'\fR 而不是 `open(log,'logfile')'\fR). 使用大寫的文件句柄既增加了可讀性, 又減少了衝突的發生。大小寫是有意義的\-\-\[lq]\s-1FOO\s0\[rq],\[lq]Foo\[rq], 和\[lq]foo\[rq] 是不同的名稱。以字母或下劃線開始的名稱可以包含數字和下劃線。 .PP 可以用一個返回相關引用的表達式來替換這樣的變量名。參見 perlref .PP 以數字開始的變量名只能包含數字。不是以字母、下劃線或數字開始的變量名只能 含有一個字符,如:$%\fR 或$$\fR. (大部分這樣的變量都有特殊的意 義。例如,$$\fR 是當前進程的id。) .SS "Context 上下文" .\".IX Subsection "Context" 在 Perl 中有時操作或值的意義取決於該操作或值所處的上下文。有兩個主要的上 下文:列表和標量上下文。相當一部分操作在需要列表的上下文中返回列表,在需 要標量的上下文中返回標量。這在有關該操作的文檔中會提到。換句話講,Perl會 重載這些操作符。英語中的某些詞,如`fish'和`sheep'與此類似。 .PP 操作可以根據不同的上下文返回不同的值。例如,如果這樣寫: .PP .nf \& int( ) .fi .PP integer 操作提供標量上下文給 <> 操作符, <> 會從STDIN 讀入一行返回給 integer 操作,然後它返回其中的整型量。但如果你這樣寫: .PP .nf \& sort( ) .fi .PP sort操作提供列表上下文給<>, <>會讀入STDIN中的每一行直到結束,然後將其傳遞給sort,sort然後將其排序輸出。 .PP 賦值比較特殊,左側的參數決定了右側的參數的上下文。賦值給標量,則右側參數的上下文是標量上下文;賦值給數組或哈希,則右側參數的上下文是列表上下文。賦值給列表(或片段,其實也是列表),右側的上下文也是列表上下文。 .PP 當你使用`use warnings'\fR 編譯指示或 Perl 的\fB\-w\fR 參數時, 你可能會看到這樣的警告:在\[lq]無效的上下文,void context\[rq] 中使用了常量 或函數。無效上下文的意思是值被丟棄不用,比如只包含有"fred"; \fR 的語句; 或是`getpwuid(0);'\fR;. 在要求列表上下文的函數 被標量上下文環境調用時,也會出現這個警告. .PP 用戶定義的子程序可能會需要查看上下文是無效,標量,還是列表。不過,大多數並 不需要這麼做。因爲標量和列表會自動插入到一個列表中。參見 perlfunc 中的 \[lq]wantarray\[rq] 以瞭解如何辨明你的函數調用時的上下文。 .SS "Scalar values 標量" .\".IX Subsection "Scalar values" Perl 中的所有數據都是標量, 標量的數組,標量的哈希. 標量可以是三種不同的值: 數字, 字符(串), 引用. 通常, 不同值之間的轉換是透明的. 雖然一個標量不可能有多個值, 但是它可以是一個包含多個值的數組或哈希的引用. .PP 標量不一定非此即彼. 不需要聲明變量的類型是"字符串","數字","引用"或其它什麼. 因爲標量會自動轉換, 所以其類型不用關心. Perl 是上下文多形語言,它的標量可以是字符串,數字或引用(包括對象). 其中字符串和數字在大多數情況下並沒有什麼不同, 引用是強化的,不可變的帶有內建引用計數和析構器的指針. .PP 標量在不是空字符串和數字0的時候被解釋爲真 TRUE. 布爾上下文是這樣一種上下文, 這時不會發生數字或字符串的自動轉換. .PP 有兩種空字符串(有時以"empty"表示), 定義了的和未定義的. 定義了的空字符串就是長度爲零的字符串,如"". 未定義的空字符串是一個值,這個值表示某事物並沒有真實值與它對應, 比如出錯, 或到達文件尾, 或者你引用一個未定義的數組或哈希的元素時,都會返回一個未定義的空字符串. 雖然在早期Perl 中,在要求已定義變量的上下文中使用未定義的變量可以使得該變量得到定義, 現在卻只有在特殊的情況下才會出現這種結果,參見the perlref manpage. 可以用defined() 函數來檢測標量是否已經定義(對數組和哈希無效),也可以用undef() 去除對變量的定義. .PP 要弄清楚一個字符串是否是有效的非0數字,只要看它是不是數字0和字母\[lq]0\[rq] 就足夠了(不過這在使用-w參數時,會顯示警告). 因爲非數字的字符串被看作0, 與awk中相似: .PP .nf \& if ($str == 0 && $str ne "0") { \& warn "That doesn't look like a number"; \& } .fi .PP 這種方法可能是最好的,因爲如若不然你不會正確對待\s-1IEEE\s0 的註釋,比如`NaN'\fR 和無窮大. 別的時候, 你可能更願意用\fIPOSIX::strtod()\fR 函數或是正則表達式來檢測字符串是否能用做數字(參見perlre). .PP .nf \& warn "has nondigits" if /\eD/; \& warn "not a natural number" unless /^\ed+$/; # rejects -3 \& warn "not an integer" unless /^-?\ed+$/; # rejects +3 \& warn "not an integer" unless /^[+-]?\ed+$/; \& warn "not a decimal number" unless /^-?\ed+\e.?\ed*$/; # rejects .2 \& warn "not a decimal number" unless /^-?(?:\ed+(?:\e.\ed*)?|\e.\ed+)$/; \& warn "not a C float" \& unless /^([+-]?)(?=\ed|\e.\ed)\ed*(\e.\ed*)?([Ee]([+-]?\ed+))?$/; .fi .PP 數組的長度是標量. 通過$#days你可以知道@days的長度. 技術上講,這不是數組的長度; 而是最後一個元素的下標,因爲第一個元素的下標是0. 對$#days 賦值會改變數組的長度. 以這種方式減少數組的話, 會破壞其中的值, 再增加其長度也不能恢復. (Perl 4中是可以的, 我們改變了它以確保析構器被及時調用.) .PP 你可以使用一些小技巧來預擴展一個數組(如果你知道它將會變得很大的話). 可以用給超出數組範圍的元素賦值的方法擴展數組. 可以給數組賦值一個空列表以清空數組. 下面語句等價: .PP .nf \& @whatever = (); \& $#whatever = -1; .fi .PP 數組處於標量上下文中時, 返回值是數組的長度. (列表在標量上下文中,返回值是列表的最後一個元素,像是C中的逗號操作符, 而內建函數的返回值由它們自己決定.) 以下語句爲真: .PP .nf \& scalar(@whatever) == $#whatever - $[ + 1; .fi .PP Perl 5 改變了$[\fR 的意義: 不必擔心別的程序改變了$[\fR 的值. (換言之,不推薦使用$[\fR ) 所以,可以寫成這樣: .PP .nf \& scalar(@whatever) == $#whatever + 1; .fi .PP 有些程序員爲了明確起見, 會使用顯式的轉換: .PP .nf \& $element_count = scalar(@whatever); .fi .PP 當哈希處於標量上下文中時, 如果哈希爲空, 返回值爲假, 如果非空, 返回值爲真; 說得更精確些, 返回值是個字符串, 由已經使用的存儲段和分配的全部存儲段組成,二者之間以斜槓分隔. 這可以用來反映Perl的哈希算法的好壞. 例如, 你的哈希中有10,000個元素,但是%HASH\fR 的標量值爲"1/16"\fR, 則說明僅用到了16個存儲段中的一個, 也許10,000個元素都在這一個存儲段中. 最好不要發生這種情況. .PP 你可以預先爲哈希分配空間, 這要使用給\fIkeys()\fR 函數賦值的方法來實現. 實際分配的空間是大於所給值的二的冪: .PP .nf \& keys(%users) = 1000; # 分配 1024 空間 .fi .SS "Scalar value constructors 標量數據構造" .\".IX Subsection "Scalar value constructors" 數值常量有以下浮點和整數格式: .PP .nf \& 12345 \& 12345.67 \& .23E-10 # a very small number \& 3.14_15_92 # a very important number \& 4_294_967_296 # underscore for legibility \& 0xff # hex \& 0xdead_beef # more hex \& 0377 # octal \& 0b011011 # binary .fi .PP 在數字常量中可以在數字間插入下劃線來增加可讀性。例如,可以三位一組 (Unix 樣式的分組,例如 0b110_110_100),或者四位一組 (來表示 nibbles,例如 0b1010_0110),或者其他分組。 .PP 字符串通常以單引號或雙引號括起. 與標準Unix shells中的引號相似: 雙引號可以接收轉義和變量; 單引號不可以 (除了`\e''\fR 和`\e\e'\fR)). C 樣式的轉義字符可以用來輸入新行,跳格等字符,轉義字符的列表可以參見 perlop 中的\[lq]Quote and Quote-like Operators\[rq] .PP 十六進制,八進制,或二進制以字符串形式表示(如:'0xff'),不能自動轉換爲十進制形式. hex() 和 oct() 函數可以實現轉換. 參見 perlfunc 中的 hex 和 oct 瞭解詳情. .PP 可以在字符串中直接加入新行. 字符串中的變量只能是標量,數組和數組或哈希的片段 (換言之, 以$或@開始, 後跟下標.). 以下語句打印``The price is $\&100.'' .PP .nf \& $Price = '$100'; # not interpolated \& print "The price is $Price.\en"; # interpolated .fi .PP perl 中沒有 double interpolation,因此$100\fR 保持不變。 .PP 正如在有些shell中一樣, 你可以用花括號括起變量名, 以便區分變量名和其後的字母及下劃線. 如果要將一個變量改寫爲字符串時,必須這樣做,以避免與後面的雙冒號或單引號連接起來,否則會被當作包名: .PP .nf \& $who = "Larry"; \& print PASSWD "${who}::0:0:Superuser:/:/bin/perl\en"; \& print "We use ${who}speak when ${who}'s here.\en"; .fi .PP 如果沒有花括號, Perl會尋找 $whospeak, $who::0, 和 $who's 變量. 後兩個是不存在的 who 包中的$0 和 $s. .PP 實際上, 花括號中的標識符必須是字符串, 哈希的下標也必須是字符串. 都不需要引號, 前面的例子$days{'Feb'} 可以寫作 $days{Feb} 引號會自動加上. 但是下標中的其它複雜內容被解釋爲表達式. .PP \fIVersion Strings\fR .\".IX Subsection "Version Strings" .PP \fB注意:\fR Version Strings (v\-strings) have been deprecated. They will not be available after Perl 5.8. The marginal benefits of v\-strings were greatly outweighed by the potential for Surprise and Confusion. .PP 類似`v1.20.300.4000'\fR 這樣的形式被解釋爲一個字符串. 這種形式稱爲 v\-strings,提供了更易讀的方法來構造字符串,比起"\ex{1}\ex{14}\ex{12c}\ex{fa0}"\fR 更加易讀. 這在表示 Unicode 字符串時很有用, 在使用字符串比較命令(`cmp'\fR,`gt'\fR,`lt'\fR 等)比較版本號時也非常有用. 如果其中的點號多於兩個, 則開始的`v'\fR 可以省略. .PP .nf \& print v9786; # prints UTF-8 encoded SMILEY, "\ex{263a}" \& print v102.111.111; # prints "foo" \& print 102.111.111; # same .fi .PP 這種形式可以用於require 和 use 中作版本檢查.\[lq]$^V\[rq] 特殊變量中的Perl版本號就是以這種形式保存的. 參見 perlvar 中的\[lq]$^V\[rq] 注意使用 v\-strings 來保存 IPv4 地址是不可移植的,除非同時使用 Socket 包的\fIinet_aton()\fR/\fIinet_ntoa()\fR 函數。 .PP 注意從 Perl 5.8.1 開始單個數字的 v\-strings (類似`v65'\fR) 如果在`=>'\fR 操作符(通常用來從 hash 值中區分開 hash 鍵) 之前,不是一個 v\-strings,而是解釋爲字符串 ('v65')。在 Perl 5.6.0 到 Perl 5.8.0 它一直是 v\-strings,但是這樣帶來了更多混淆和錯誤而不是優點。多個數字的 v\-strings,類似`v65.66'\fR 和65.66.67\fR,繼續總是被當作 v\-strings .PP \fI特殊常量\fR .\".IX Subsection "Special Literals" .PP 特殊變量 _\|_FILE_\|_, _\|_LINE_\|_, 和 _\|_PACKAGE_\|_ 代表當前文件名,行號,和包名. 它們只能作爲單獨的符號來使用; 不能用於字符串中內插. 如果沒有當前包(用`package;'\fR 指令來實現), 則_\|_PACKAGE_\|_ 是一個未定義的值. .PP 控制字符 ^D 和 ^Z, 以及 _\|_END_\|_ 和 _\|_DATA_\|_ 變量可以表示文件的邏輯結束. 其後的文本被忽略. .PP _\|_DATA_\|_ 之後的文本可以通過文件句柄`PACKNAME::DATA'\fR 讀取,`PACKNAME'\fR 是 _\|_DATA_\|_ 所在的包的名稱. 句柄指向_\|_DATA_\|_ 後面的文本. 讀取結束程序會自動關閉該句柄`close DATA'\fR. 爲了與 _\|_DATA_\|_ 還沒有出現以前已經存在的程序兼容, _\|_END_\|_ 在頂級腳本中與 _\|_DATA_\|_ 性質相同(在用`require'\fR 或`do'\fR 調用時是不同的) 不過可以通過`main::DATA'\fR 來調用其中的內容. .PP 參見 SelfLoader 詳細瞭解 _\|_DATA_\|_, 其中還有例子. 要注意在BEGIN 塊中無法讀取DATA句柄: 因爲BEGIN 塊在編譯時即被執行, 而此時 _\|_DATA_\|_ (或 _\|_END_\|_) 還未被程序看到. .PP \fI裸詞\fR .\".IX Subsection "Barewords" .PP 在文法上沒有特殊意義的詞語都被看作字符串. 稱之爲 "裸詞". 和文件句柄以及標籤一樣, 僅包含小寫字母的裸詞有可能在將來與程序中的保留詞發生衝突, 實際上,當你使用`use warnings'\fR 語句,或是\fB\-w\fR 選項時, Perl會對此提出警告. 一些人可能希望完全禁止這樣的詞. 如果有如下語句: .PP .nf \& use strict 'subs'; .fi .PP 那麼不能被解釋爲子程序的裸詞會引起編譯時錯誤. 這種限制到塊結束時終止. 而內部的塊可以撤消這一限制, 用`no strict 'subs''\fR .PP \fI數組合並分隔符\fR .\".IX Subsection "Array Joining Delimiter" .PP 數組和序列被合併爲雙引號引用的字符串時,以變量$"\fR 指定的值 (如果指定了\[lq]use English;\[rq] 那麼是$LIST_SEPARATOR\fR 的值) 作爲分隔符,默認是空格。下列語句等價: .PP .nf \& $temp = join($", @ARGV); \& system "echo $temp"; .fi .PP .nf \& system "echo @ARGV"; .fi .PP 在搜索模式中(在雙引號字符串中也是)有一個易混淆之處:`/$foo[bar]/'\fR 應該是`/${foo}[bar]/'\fR (`[bar]'\fR 是正則表達式的字符類) 還是`/${foo[bar]}/'\fR/ (`[bar]'\fR 是數組@foo\fR 的下標) 呢? 如果@foo\fR 不存在, 那很明顯它應該是字符類. 如果@foo\fR 存在, Perl 會盡力猜測`[bar]'\fR 的含義, 且它幾乎總是對的. 如果它猜錯了, 或者你比較偏執, 你可以使用花括號. .PP here\-document 的語法已經被移動到 perlop 中的\[lq]Quote and Quote-like Operators\[rq] .SS "List value constructors 列表值構造" .\".IX Subsection "List value constructors" 列表是用逗號分開的各個值組成的(如果優先級需要的話,外面還要用圓括號包圍): .PP .nf \& (LIST) .fi .PP 在不需要列表的上下文中, 列表的值是最後一個元素的值, 這與C中的逗號操作符類似. 例如: .PP \& @foo = ('cc', '-E', $bar); .PP 將列表賦給數組@foo, 但是 .PP \& $foo = ('cc', '-E', $bar); .PP 將$bar 的值賦給$foo. 注意, 數組在標量上下文中的值是數組的長度; 下例將3賦給$foo: .PP .nf \& @foo = ('cc', '-E', $bar); \& $foo = @foo; # $foo gets 3 .fi .PP 列表的最後可以輸入逗號, 所以這樣也是正確的: .PP .nf \& @foo = ( \& 1, \& 2, \& 3, \& ); .fi .PP 要將here-document 賦給數組, 一行作爲一個元素, 可以這樣作: .PP .nf \& @sauces = <'\fR 操作符會更易讀.`=>'\fR 與逗號作用相同, 不過它 還有一個作用, 那就是可以使它左側的對象被解釋爲字符串: 如果該對象是裸 字的話,將是合法的標識符 (`=>'\fR 不引用包含雙冒號的複合標識符). 這在初始 化哈希時棒極了: .PP .nf \& %map = ( \& red => 0x00f, \& blue => 0x0f0, \& green => 0xf00, \& ); .fi .PP 或者初始化哈希的引用: .PP .nf \& $rec = { \& witch => 'Mable the Merciless', \& cat => 'Fluffy the Ferocious', \& date => '10/31/1776', \& }; .fi .PP or for using call-by-named-parameter to complicated functions: .PP .nf \& $field = $query->radio_group( \& name => 'group_name', \& values => ['eenie','meenie','minie'], \& default => 'meenie', \& linebreak => 'true', \& labels =>\e%labels \& ); .fi .PP 注意哈希初始化時的順序和輸出時的順序並不一定相同. 要得到順序的輸出可以參見 perlfunc 中的\[lq]sort\[rq] .SS "Subscripts 下標" .\".IX Subsection "Subscripts" 數組可以用一個美元符號,加上它的名字(不包括前導的`@'\fR),加上方括號和其中包含的下標來取得值。例如: .PP .nf \& @myarray = (5, 50, 500, 5000); \& print "Element Number 2 is", $myarray[2], "\en"; .fi .PP 數組下標從 0 開始。負值下標返回從尾部開始數的值。在我們的例子中,$myarray[\-1]\fR 將是 5000,$myarray[\-2]\fR 是 500。 .PP Hash 下標與此類似,但是不使用方括號而是花括號。例如: .PP .nf \& %scientists = \& ( \& "Newton" => "Isaac", \& "Einstein" => "Albert", \& "Darwin" => "Charles", \& "Feynman" => "Richard", \& ); .fi .PP \& print "Darwin's First Name is ", $scientists{"Darwin"}, "\en"; .SS "Slices 片段" .\".IX Subsection "Slices" 通常對哈希或數組一次訪問一個元素. 也可以使用下標對列表元素進行訪問. .PP .nf \& $whoami = $ENV{"USER"}; # one element from the hash \& $parent = $ISA[0]; # one element from the array \& $dir = (getpwnam("daemon"))[7]; # likewise, but with list .fi .PP 片段可以一次訪問列表,數組或哈希中的幾個元素, 這是通過列表下標來實現的. 這比分別寫出每個值要方便一些. .PP .nf \& ($him, $her) = @folks[0,-1]; # array slice \& @them = @folks[0 .. 3]; # array slice \& ($who, $home) = @ENV{"USER", "HOME"}; # hash slice \& ($uid, $dir) = (getpwnam("daemon"))[2,7]; # list slice .fi .PP 既然可以給列表賦值, 當然也可以哈希或數組的片段賦值. .PP .nf \& @days[3..5] = qw/Wed Thu Fri/; \& @colors{'red','blue','green'} \& = (0xff0000, 0x0000ff, 0x00ff00); \& @folks[0, -1] = @folks[-1, 0]; .fi .PP 上面的操作與下列語句等價: .PP .nf \& ($days[3], $days[4], $days[5]) = qw/Wed Thu Fri/; \& ($colors{'red'}, $colors{'blue'}, $colors{'green'}) \& = (0xff0000, 0x0000ff, 0x00ff00); \& ($folks[0], $folks[-1]) = ($folks[-1], $folks[0]); .fi .PP 既然改變片段就會改變數組或哈希的原始值, 那麼`foreach'\fR 結構可以部分或全部地改變數組或哈希的值. .PP \& foreach (@array[ 4 .. 10 ]) { s/peter/paul/ } .PP .nf \& foreach (@hash{qw[key1 key2]}) { \& s/^\es+//; # trim leading whitespace \& s/\es+$//; # trim trailing whitespace \& s/(\ew+)/\eu\eL$1/g; # "titlecase" words \& } .fi .PP 空列表的片段還是空列表, 因此: .PP .nf \& @a = ()[1,0]; # @a has no elements \& @b = (@a)[0,1]; # @b has no elements \& @c = (0,1)[2,3]; # @c has no elements .fi .PP 但是: .PP .nf \& @a = (1)[1,0]; # @a has two elements \& @b = (1,undef)[1,0,2]; # @b has three elements .fi .PP 下例利用了這一特性,當返回空列表時循環終止: .PP .nf \& while ( ($home, $user) = (getpwent)[7,0]) { \& printf "%-8s %s\en", $user, $home; \& } .fi .PP 我們在前面說過, 標量上下文中的列表賦值返回值是右側的元素個數. 空列表沒有元素, 所以當口令文件讀完後, 返回值是0而不是2. .PP 爲什麼對哈希的片段使用'@'而不是'%'呢. 因爲括號的類型(方括號或花括號)決定了它是數組還是哈希. 而數組或哈希的開始字符('$'或'@')表示返回值是單個值還是多個值(列表). .SS "Typeglobs and Filehandles 全局類型和文件句柄" .\".IX Subsection "Typeglobs and Filehandles" Perl 使用叫做 全局類型 的類型來支持整個符號表項. 全局類型的前綴是*, 因爲它表示所有的類型. 這在過去通常用來給函數傳遞數組或哈希的引用, 但是現在有了真正的引用, 這就幾乎不需要了. .PP 現在,全局類型的主要用途是創建符號表別名. 如下賦值: .PP \& *this = *that; .PP 使得$this 成爲 $that的別名, @this 成爲 @that的別名,%this 成爲 %that的別名, &this 成爲 &that的別名, 等等. 使用引用會更安全. 這樣: .PP \& local *Here::blue =\e$There::green; .PP 暫時使 $Here::blue 成爲 $There::green的別名, 但不會使 @Here::blue 成爲 @There::green的別名, 也不會使 %Here::blue 成爲 %There::green的別名, 等等. 參見 perlmod 中的 Symbol Tables 有多個例子. 看起來可能有些怪異, 不過這卻是整個import/export系統的基礎. .PP 全局類型的其它用途還有, 給函數傳輸文件句柄或是創建新的文件句柄. 如果你要使用全局類型代替文件句柄, 可以這樣做: .PP \& $fh = *STDOUT; .PP 或者使用真正的引用, 象這樣: .PP .nf \& $fh =\e*STDOUT; .fi .PP 參見 perlsub 有關於間接句柄的多個例子. .PP 全局類型也是使用local() 創建局部文件句柄的一種方法. 作用範圍在當前塊之內, 但是可以被傳回.例如: .PP .nf \& sub newopen { \& my $path = shift; \& local *FH; # not my! \& open (FH, $path) or return undef; \& return *FH; \& } \& $fh = newopen('/etc/passwd'); .fi .PP 既然我們有*foo{THING} 這樣的記法, 全局類型不再多用於文件句柄,但在從函數傳出或向函數傳入新的文件句柄時它還是必需的.因爲*HANDLE{IO} 只有在HANDLE 已經是文件句柄時才起作用. 換言之, 在建立新符號表項時必須使用 *FH; *foo{THING} 是不行的. 不知道該用誰時, 使用 *FH .PP 所有能創建文件句柄的函數 (open(), opendir(), pipe(), socketpair(), sysopen(), socket(), 和 accept()) ,在傳遞給它們的句柄是標量時,會自動創建一個匿名句柄. 這使得象open(my $fh, ...) 和 open(local $fh,...) 這樣的結構可以創建一個在超出範圍時可以自動關閉的句柄,如果沒有另外的對它們的引用的話. 這大大減少了全局類型的使用,當需要打開一個可以到處使用的句柄時, 可以這樣做: .PP .nf \& sub myopen { \& open my $fh, "@_" \& or die "Can't open '@_': $!"; \& return $fh; \& } .fi .PP .nf \& { \& my $f = myopen("; \& # $f implicitly closed here \& } .fi .PP 注意如果使用了初始化的標量,那麼結果會有不同:`my $fh='zzz'; open($fh, ...)'\fR 與`open( *{'zzz'}, ...)'\fR 等價。`use strict 'refs''\fR 禁止了這樣做。 .PP 另一個創建匿名句柄的方法是用Symbol 模塊或IO::Handle 模塊或諸如此類的東西. These modules have the advantage of not hiding different types of the same name during the local(). 在 open() in the perlfunc manpage 的文末有個例子.(譯者注:說實話,對匿名句柄我現在也是一頭霧水,翻譯的不當之處,請高手指出.) .SH "SEE ALSO 參見" .\".IX Header "SEE ALSO" 參見 the perlvar manpage 瞭解 Perl的內建變量和合法變量。參見the perlref manpage, the perlsub manpage, 和 Symbol Tables in the perlmod manpage 瞭解全局類型和 *foo{THING} 語法。 .SH "中文版維護人" .B redcandle .SH "中文版最新更新" .B 2001年12月4日星期二 .SH "中文手冊頁翻譯計劃" .B http://cmpp.linuxforum.net .SH "跋" .br 本頁面中文版由中文 man 手冊頁計劃提供。 .br 中文 man 手冊頁計劃:\fBhttps://github.com/man-pages-zh/manpages-zh\fR