加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 大数据 > 正文

Perl基础知识(备忘)

发布时间:2020-12-16 00:04:58 所属栏目:大数据 来源:网络整理
导读:zz :?http://blog.csdn.net/dongtingzhizi/article/details/13509079 前两天项目中遇到了一个Perl脚本程序,需要读懂该程序,由于以前重来没有用过Perl语言,所以没法搞定。今天抽空把该语言的基础看了一遍,基本上内读懂Perl脚本程序了吧。真是如网上很多分

zz :?http://blog.csdn.net/dongtingzhizi/article/details/13509079

 前两天项目中遇到了一个Perl脚本程序,需要读懂该程序,由于以前重来没有用过Perl语言,所以没法搞定。今天抽空把该语言的基础看了一遍,基本上内读懂Perl脚本程序了吧。真是如网上很多分享的经验所说,第一:会一门语言后,学习其他的语言也就容易了,所有的语言包含的内容都差不多,只要熟悉一下该语言的语法就OK了;第二:会一门语言的有经验的开发者,跟本没有必要花大量的时间把一门新语言重头读到尾来学习他,看看基础就可以用了,用的过程中不明白的再去查阅手册或资料。

  网上不少人说Perl语言受关注度越来越低,在服务器端慢慢的被PHP取代,但是作为系统脚本语言还是有他的特点的,我觉得最主要的是:强大的正则表达式和模式匹配功能,在sed和awk中也用到了类似的语法。

  入门可以参考以下教程:

  http://www.cbi.pku.edu.cn/chinese/documents/perl/index.htm

  http://www.phpchina.com/resource/manual/perl/perl5-1.htm

? ? ? ??下面是我学习中记下的一些基础知识,当作备忘吧。



【数组】


● 首先说一种数据类型,将“列表”,例如:(1,2,3),(1,"hello",$name,10)。

● 连续数值的列表有缩写形式,例如:(1 .. 10),(1,3,5 .. 10);浮点型的(0.5 .. 3.8)表示(0.5,1.5,2.5,3.5); 字符型的(a .. z)表示小写字母a到z。

●?数组变量以@开头,可以定义与简单变量相同的名字,$var和@var不冲突。

●?数组可以直接给普通变量赋值,例如:
? ?@arr = (1 .. 3);
? ?($a,$b) = @arr; ? ? ? ??#a为1,b为2
? ?($c,$d,$e,$f) = @arr; ? ? ? ??#c为1,d为2,e为3,f为null

●?qw函数创建字符串数组,不需要使用逗号和引号,例如:
? ?@arr = qw(aaa bbb ccc ddd);

●?可以从标准输入读入数组,按CTRL+D结束输入,例如:
? ?@arr = <STDIN>;

●?访问数组的某个元素,用$,数组下标从0开始,例如:
? ?@arr = (1 .. 3);?

? ?$arr[0]; $arr[1]; $arr[2]; 分别访问数组第1、2、3个元素。

● ?获取数组长度:$len = @arr;

● ?数组截取(用@),(注意规则是:取多个数组元素用@,取单个数组元素用$。)例如:
? ? ? ? @arr = (1 .. 6);
? ? ? ??@sub1 = @arr[0,4];? ? ? ??#sub1为(1,5)
? ? ? ??@sub2 = @arr[1,5];? ? ? ??#sub1为(2,4,6)
? ? ? ??@sub1 = @arr[0,3..5];? ? ? ??#sub1为(1,5,6)

● ?数组访问支持负数下标,例如:
? ? ? ??@arr = (1,3);
? ? ? ??$item = $arr[-1];? ? ? ??#item为3
? ? ? ??$sub = @arr[0,-1];? ? ? ??#sub为(1,3)

(作用于数组的函数)

● ?chop和chomp函数可用于数组,例如:chomp(@arr)用于去掉arr中所有字符串尾部的换行符。

● ?push和pull操作数组相当于栈操作,数组的右边界为栈的顶部,即从右边“弹出”或者“压入”元素。

● ?sort用于对数组元素进行排序,调用形式为:@sorted = sort(@arr); 执行sort函数后原数组不会改变,所以要通过返回值接收排好序的数组。默认情况下,按所有元素的ASCII顺序排序(包括数字),所以不能简单调用sort函数对数值型元素排序。如果需要按数值大小进行排序,应该这样调用:
升序:@sorted = sort { $a <=> $b } @arr;
降序:@sorted = sort { $b <=> $a } @arr;

● ?reverse函数用于反转一个数组,例如:
? ? ? ??@arr = (1 .. 6);
? ? ? ??@r_arr = reverse(@arr);? ? ? ??#r_arr为(6,1)

● ?join函数用于连接一个数组的所有元素,组成一个字符串,调用形式为:$string = join("seperator",@arr,...)。例如:
? ? ? ??@arr = (aa,bb,cc);
? ? ? ??@str1 = join("-",@arr);? ? ? ??#str1为"aa-bb-cc"
? ? ? ??@str2 = joint(":",100);? ? ? ??#str2为"aa:bb:cc:hello:100"

● ?split函数与join正好相反,调用形式为:@arr = split("seperator",$str)。



【散列表(hash表,关联数组)】


● ?散列变量以%开头,例如:
? ? ? ??%hash = (a,1,b,c,3); ? #表示a对应的值为1,b对应的值为2,c对应的值为3

● ?访问散列表元素,例如:
? ? ? ??$hash{a};? ? ? ??#访问键a的值,结果为1

● ?添加新元素,例如:
? ? ? ??$hash{d} = 100;? ? ? ??#添加键d的值为100

● ?例子:

? ? ? ??@fruit =(“apple”,“banana”,“orange”,12),获取元素或修改元素值:$dict{“bananas”},$dict{“bananas”}=1。

上面的关联数组初始化也可以这样:@fruit = (“apple”=>1,“banana”=>2,“orange”=>12)。

● ?删除元素,例如:
? ? ? ??delete $hash{c};#结果%hash为:(a,d,100)

● ?判断散列表是否为空,例如:
? ? ? ??if (%hash)

● ?取散列表中所有键的函数:keys,例如:
? ? ? ??@keys = keys(%hash);? ? ? ??#@keys为:(a,d)

● ?取散列表中所有value的函数:values,例如:
? ? ? ??@values = values(%hash);? ? ? ??#@values 为:(a,d)

● ?循环读取所有的key-value对的函数:each,相当于c++中的关联容器的迭代器,例如:
? ? ? ??while (@item = each(%hash)) {
? ? ? ??? ? ? ??...
? ? ? ??}



【控制结构】


● ?if语句与C语言类似,不同的是C语言中的else if应该写成elsif。

● ?unless语句使用方式与if相同,但是语义与if语句相反,当条件不成立时执行then语句块。unless也有unless {} else {},当条件成立时执行else语句块。

● ?while语句与C语言类似。

● ?do-while语句与C语言类似。

● ?until语句使用方式与while相同,但是语义与while语句相反,当条件不成立时继续循环,当条件成立时结束。

● ?for语句与C语言类似。一个特别的例子:循环变量递增语句后可以用逗号表达式增加其他执行语句。
? ? ? ?for ( $a = 1; $a < 10; $a++,print "the value of a is : $a=n" ) {
? ? ? ?? ? ? ?......
? ? ? ?}

● ?foreach语句非常实用,可以*自动*依次获取数组变量中的每一个元素来执行循环,格式如下:
? ? ? ? foreach 变量 (数组变量) {


? ? ? ? }
? ? ? ? foreach $var (@arr) {


? ? ? ? }

● ?循环控制语句有三种:next,last和redo,next相当于C语言中的continue;last相当于C语言中的break;而redo表示不进行循环条件的判断,再执行依次循环体,应该较少适用吧。



【函数】


● ?函数定义以sub开头,格式如下:
? ? ? ? sub NAME {
? ? ? ??? ? ? ??code;
? ? ? ??}

● ?函数可以接收调用者传入的参数,参数通过数组@_传入,所以可以读取该数组来获取每一个参数。

● ?通常函数调用方法:&NAME。

● ?函数通过return语句返回一个值。

● ?看一个例子吧:
? ? ? ??sub add {
? ? ? ??? ? ? ??my?$res = $_[0] + $_[1];
? ? ? ??? ? ? ??return $res;
? ? ? ??}
? ? ? ??$a = 1; $b = 2;
? ? ? ??$c = &add($a,$b);
? ?参数获取也可以使用对参数数组@_使用shift函数来实现,上面的add函数可以写成:
? ? ? ??sub add {
? ? ? ??? ? ? ??my?$a1 = shift(@_);
? ? ? ??? ? ? ??my?$a2 = shift(@_);
? ? ? ??? ? ? ??my?$res = $a1 + $a2;
? ? ? ??? ? ? ??return $res;
? ? ? ??}

● ?预定义子程序,如BEGIN和END函数,BEGIN函数就是在整个程序执行前被调用的函数,END是在整个程序结束后被调用的函数。其他先不深究了,用到的时候再说吧。

● ?函数内部可以定义临时数据(C++中称为局部变量),可以使用local和my两个函数来定义临时数据,二者是有一定区别的,据说local还有很深奥的使用方法,这里也不深究了。local和my有一个最大区别是,local定义的临时数据会传递到嵌套调用的下一个函数中,而my定义的临时数据值在当前域中有效,也就是说:my定义的临时数据才是C++中通常意义的局部变量,所以对二者不是很清楚的话,使用my是更加保险滴!



【文件和目录操作】


● ?打开与关闭文件

? ? ? ?open(HANDLE,"filename"); ? ? ? ?#获取读文件句柄 ?
? ? ? ?open(HANDLE,">filename"); ? ? ? ?#获取写文件句柄,若文件存在则先清空内容再写 ?

? ? ? ?open(HANDLE,">>filename"); ? ? ? ?#获取写文件句柄,若文件存在则在原内容之后追加

? ? ? ?close(HANDLE); ? ?#关闭文件,参数为文件句柄

● ?更高级的打开文件方式

? ? 为了保证可靠性,一般打开文件后需要判断打开文件是否成功,Perl中一般会这样写:

? ? ? ? open(HANDLE,...) || die ("Could not open file ***!n$!");?
? ? ? ??open(HANDLE,...) || warn ("Could not open file ***!n$!");?

? ? 二者的区别是,die会结束运行,warn只是输出警告信息然后继续往后执行。

? ? $!是编译器给出的错误信息,看看下面的例子,假如打开一个不存在的文件test.txt:

? ??? ??open(HANDLE,”test.txt“) || die ("Could not open file test.txt!n"); ? ?

? ? 会输出:

? ? ? ??Could not open file test.txt!

? ? 如果架上$!,如下:?

? ? ? ?open(HANDLE,”test.txt“) || die ("Could not open file test.txt!n$!");?

? ? 则会输出:

? ? ? ??Could not open file test.txt!

? ? ? ? No such file or directory at ./3.pl line 5. (该行是编译器给出的错误信息)

? ? 可以看出,这种打开文件的方法非常简练使用,所以差不多就成了标准写法吧。

● ?读取文件

? ??$a = <HANDLE>; ? ?#读取文件中的一行

读取后,文件指针会自动后移到下一行,因此可以用在循环中按行循环读取一个文件,例如:

? ?while ( $line = <HANDLE> ) {
? ? ? ? print $line;
? ? }

? ??@a = <HANDLE>; ? ?#将文件整个赋给数组@a,数组的每个元素对应文件的一行?

详见:http://blog.csdn.net/taesimple/article/details/6307639

● ?打开/读取/关闭目录

? ? 打开/关闭目录与操作文件类似,看下面的例子,读取该目录中所有的子目录和文件,做不同的处理:

??? opendir($handle,$path);
?? ?while($item = readdir($handle)) {
?? ??? ?if(-d $item && !($dic =~ m/.$|..$/)) {
?? ??? ??? ...
?? ??? ?}
??????? if ( -f $item ) {
?????????? ...
??????? }
??? }
? ? closedir($handle);

● ?关于当前目录

这是非常关键的,因为如果不是使用全局路径来操作文件或目录,都是相当于当前目录的,所以必须保证当前目录正确。

(获取当前目录)

有两种方法:
(1)
? ? ? ?use Cwd;
? ? ? ?my $dir = getcwd;
$dir中即为当前目录的完整路径信息。
(2)
? ? ? ?my $dir = $ENV{'PWD'};
#ENV是一个散列,用于存放环境变量。$PWD是Linux的环境变量,表示当前所在目录。

(改变当前目录)

用chdir函数。

● ?测试文件或目录

? ? 使用”-x 文件或目录名“的格式来测试文件或者目录的状态,-x文件测试操作符有很多不同的参数,常用的有:

? ? -e: 是否存在

? ? -d: 是否为目录

? ? -f: 是否为普通文件

? ? -r: 是否可读

? ? -w: 是否可写

? ? -x: 是否可执行

? ? -l: 是否为符号链接

? ? 例如下面的例子表示,是否存在”test.txt"文件:

? ? ? ? if ( -e "test.txt" )

● ?其他文件操作函数

rename移动文件或者改名;unlink删除文件;chmod改变文件属性。

● ?其他目录操作函数

mkdir创建目录,rmdir删除空目录。



【正则表达式】


● ?基本模式:$str =~ /abc/表示在字符串变量$str中匹配"abc",$str !~ /abc/表示在字符串变量$str中*不*匹配"abc"(即不含“abc”)。匹配表达式可以返回真假结果值,例如:$match = ($str =~ /abc/),如果匹配成功则$match为真,否则为假,也可以作为条件判断,例如:if ($match = ($str =~ /abc/)?) 或者直接用匹配表达式判断: if ($str =~ /abc/?)。

● ?特殊符号:*,?,.(含义不赘述了),其中“.”表示匹配*除了换行符以外*的任意一个字符。

● ?关于转义处理,上面的特殊符号(*/?/.)有时候需要转义处理,因为这些特殊符号出现在匹配串中时有特殊含义,如果真的需要匹配这些字符时就要转义处理,例如要匹配“hello.”时,应该这样写:$str =~ /hello./,如果不进行转义处理写成$str =~ /hello./的话表示,匹配hello后面跟上任意一个非换行字符。

如果特殊符号(*/?/.)出现在[ ]中则不用转义,例如要匹配:“hello.”和“hello!”,应该这样写:$str =~ /hello[.!]/,其中的"."不用进行转义处理。

● ?锚定模式:b匹配单词边界,B匹配非单词边界,^和$匹配整个字符串的头部和末尾。

● ?排除符号:^,跟匹配字符串头部的符号是一样的,但是只出现在[ ]中,表示“非“的含义,?例如:$str =~ /hello[^0-9abc]/表示hello后面紧接着的字符不是0-9或a或b或c。

● ?管道符号:|,表示”或“的含义,可以将需要匹配的候选模式用”或“连接起来,例如:$str =~ /hello | hi/表示匹配”hello”或者匹配“hi"。

● ?保存匹配片段,可以将匹配模式中的全部或者部分用括号扩起来,然后在匹配模式中引用或者在匹配完成后引用。

? ? ? ?匹配模式中引用时使用1、2...,例如:

? ? ? ?$str =~ /(d{3})-1/,解释:d{3}表示连续的3个数字,用括号括起来表示一个保存的匹配片段,1表示引用第一个匹配片段(即第一个括号里匹配的内容,匹配片段从左至右以1开始按顺序编号),所以该匹配模式匹配形如”123-123”、“010-010“的串。

? ? ? ?$str =~ /(w)(w)-12/,匹配的串例如:"ab-ab","a1-a1","aa-aa"等。

? ? ? ?匹配结束后引用使用$1、$2...,例如:

? ? ? ?if ( $res = $str =~ /([a-z]+)-([a-z]+)/ ) {

? ? ? ? ? ? ? print "str1 : $1n";

? ? ? ? ? ? ? print "str1 : $2n";

? ? ? ?} ? ?#如果成功匹配的话,将匹配的两个匹配片段打印出来。

● ?特殊符号的优先级:{ + * ? ( ) } 大于 { ^ $ b B } 大于 | 。

● ?匹配选项:i,使用方式:/.../i,表示匹配时忽略大小写,例如:$str =~ /hello/i 可以成功匹配"hello",”HELLO","hELlo"等等。

●?匹配选项:g,使用方式:/.../g,表示匹配所有的串,默认情况下(不加g)一次匹配成功后便结束,但是有时候需要获取所有的匹配结果。例如串$str = "Today is 28 Nov,2012,I am 20 years old.",现在需要获取其中所有的数,显然以前的匹配方式 $str =~ /bd+b/只能匹配第一个数(28),如果要获取所有的数应该写成:$str =~ /bd+b/g,但是当有多个匹配结果时,怎样获取结果呢?有两种方式,如下:

第一种:用while循环获取(代码如下),只要能成功匹配一次,while判断就为真,而且匹配表达式能自动的往后继续匹配。用$&获取匹配的结果($&通常包含满足最近匹配的字符串)。

#! /usr/bin/perl

$str = "This is an example.";
$match = "b[a-z]+b";
while ( $str =~ /$match/gi ) {
? ? print "match : $&n";
}

第二种:直接用数组一次获取匹配表达式的所有结果(代码如下),@res存储了所有成功匹配的字符串。

#! /usr/bin/perl

$str = "This is an example.";
$match = "b[a-z]+b";
@res = ( $str =~ /$match/gi );?
foreach $item (@res) {
? ? print "match : $itemn";
}

● ?搜索并替换,格式为:$str =~ s/源串/替换串/。例如:$str =~ s/yes/no/,将串中的yes换成no。例如$str = "yes,great!",执行结果则为"no,great!"。

替换选项:g。如果$str = "yes,great! yes,wonderful!",则结果为“no,wonderful!”,说明跟匹配是一样的,只替换第一个匹配串,如果需要替换所有匹配的串,可以用前面提到的g选项,所以可以写成:$str =~ s/yes/no/g,表示替换所有匹配串。

替换选项:i。表示忽略大小写,与匹配选项i是同样的意义,略。

表达式选项:e。表示替换串中支持可以计算的表达式。举个简单的例子,$str =~ s/the/9 * 9/ge表示替换所有的“the"为表达式“9 * 9“的结果,而不是替换为字符串”9 * 9",所以对于$str = "the dog is here!"执行结果为“81 dog is here!"。

● ?搜索并翻译,用的比较少吧,通过下面的简单例子很好理解。

?注意替换操作符s/string1/string2/和翻译操作符tr/string1/string2/的区别,前者查找string1并替换成string2,后者查找string1中的第一个字符替换为string2中的第一个字符,查找string1中的第二个字符替换为string2中的第二个字符,以此类推,也就是tr操作符不把string1和string2当做字符串而是一个个独立的字符。下面的例子显示了二者的区别。
? ? ? ? $str = “abc123cba”;
? ? ? ??$str =~ s/abc/xyz/; ?#结果str为:”xyz123cba”
? ? ? ??$str = “abc123cba”;
? ? ? ??$str =~ tr/abc/xyz/; ?#结果str为:”xyz123zyx”



【其他杂项】


● ?print可以加括号也可以不加。

● ?perl特有的指数运算符**,如2**3。

● ?chop截掉尾部的任意字符,而chomp只截掉换行符,一般处理STDIN的输入时用chomp函数,例如:

? ? ? ?chomp($input = <STDIN>);

● ?去掉标准输入的行尾换行符的另外一种方式,例如:

? ? ? ?$in = <STDIN>;
? ? ? ?print (“Hello world!n”) if ($in eq “yes”);

终端输入yes,结果没有输出,这是因为$in末尾有换行符,可以用模式匹配删除掉,如下:

? ? ? ?$in = <STDIN>;
? ? ? ?$in =~ s/[rn]//;
? ? ? ?print (“Hello world!n”) if ($in eq “yes”);

● ?字符串比较与数值比较:

字符串比较用:lt,gt,eq,le,ge,ne,cmp(比较,返回1,or-1)。数值比较用:<, >,==,<=, >=,!=,<=>(比较,返回1,or-1)。

需要注意的是,不能用数值比较符号去比较字符串,因为用数值符号比较时,会先把左右操作数转换为数值类型,非数字字符串转换结果为0。所以,例如:if (“abc”== “yes”) 判断结果为真!

● ?比较运算符a <=> b,a大返回1,b大返回-1,相等返回0。

● ?多行注释:
? ? ? ? =pod
? ? ? ??codes to comment
? ? ? ??=cut
? ? 注意:=pod和=cut只能在行首,以=开头,以=cut结尾。

● ?字符串连接用'.'运算符,例如:"hello"." "."world",结果为"hello world"。

● ??"abc"x3表示"abc"串重复3次,结果为"abcabcabc"。

● ??普通变量“$”开头,数组变量“@”开头,关联数组“%”开头,另外访问数组或者关联数组的单个元素时以“$”开头。数组元素访问用“[]”,关联数组元素访问用“{}”,例如:

数组:@arr =(1,3),访问第三个元素:$arr[2];

关联数组:@fruit =(“apple”,“orange”=>12)。

● ??使用正则表达式进行模式匹配时,$&为上一次匹配成功的字符串。

● ??去掉行首或行尾的空白符(包括换行符)的正则表达式写法:
? ? ? ?$inputline = <STDIN>;
? ? ? ?$inputline =~ s/^s+|s+n$//g;

● ??使用正则表达式进行模式匹配时,$&为上一次匹配成功的字符串。

● ??去掉行首或行尾的空白符(包括换行符)的正则表达式写法: ? ? ? ? $inputline = <STDIN>; ? ? ? ??$inputline =~ s/^s+|s+n$//g;

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读