本文主要是介绍Perl语言入门读书笔记 | 四. 子程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
其实就是所谓的函数。
1. 子程序基础
- 子程序中可以使用任何全局变量;
- 子程序最后一次运算的结果,会被自动当成子程序的返回值。当然,也可以使用return关键字。
sub max{if($_[0] > $_[1]){ $_[0] }else{ $_[1] }
}print &max(3, 5);
$_[0]和$_[1]、@_ 都是函数的私有变量,不会受外界影响。默认情况下,Perl里面的所有变量都是全局变量,哪怕是在函数里创建的。这是一件很危险的事。如果不小心又创建了一个之前出现过的同名变量的话,很容易在无意间就改变了之前出现过的这个变量的值。通过my关键字,可以创建私有变量,和全局变量两不相犯,从而避免了这种情况。
sub max{if(@_ != 2){ # 判断参数个数print "Error\n"}my($m, $n) = @_; # 创建新的私有变量,并将参数赋值给变量if($m > $n){ $m }else{ $n }
}print &max(3, 5);
一个改进的、可以接受任意个参数的max函数如下,这个算法的名字很有意思。“高水线(high-watermark)”:大水过后,在最后一波浪消退时,高水线会标示出所见过的最高水位。
sub max{my($max_so_far) = shift @_; # 数组中的第一个值,暂时当它最大foreach (@_){if($_ > $max_so_far){$max_so_far = $_;}}$max_so_far;
}print &max(3, 5, 10, 4, 6);
2. my关键字
my关键字不会更改变量赋值时的上下文:
my ($num) = @_; # 列表上下文,和($num) = @_; 相同, $num被设为数组的第一个数
my $num = @_; # 标量上下文,和 $num = @_; 相同, $sum被设为数组的大小
在日常编程中,最好将每个新变量都使用my声明,让它保持在自己所在的词法作用域内。
foreach my $rock (qw{a b c}){print $rock;
}
use strict关键字:编译器会强制使用一些严格的、良好的编程风格。比如,变量要先声明(用my关键字),后使用:
use strict;
my $vivi = 3;
$vivi += 1;
以上代码没有加my的话,就会报错:
Global symbol “$vivi” requires explicit package name (did you forget to declare “my $vivi”?)
3. 省略函数调用的$号
在不加$号也不会产生歧义的情况下,函数调用时函数名前的$号是可以省略的。如下面这种情况,函数shift不仅可以省略$号,连参数列表两边的括号都可以省略:
use strict;
my @array = qw(a b c);
my @dealed = shift @array;
如果自己定义的子程序与Perl内置函数同名的话,为了调用自己定义的子程序,就必须加$号。所以更一般的情况是,调用自建函数时务必使用$号,除非你很确定Perl所有的内置函数名没有与你的自建函数同名。
4. state关键字
sub marine{$n += 1;print "$n\n"
}&marine;
&marine;
这段代码输出1和2,$n是全局变量。但是加上use strict;后,这段代码将出错,因为$n没有被定义。但是如果将$n定义为my,每次函数执行完成后,$n的内容会被清空,输出结果就会变成1和1。为了保留函数结果,我们使用state关键字来声明持久性私有变量。
use strict;
use feature 'state';
sub marine{state $n = 0;$n += 1;print "$n\n";
}&marine;
&marine;
第一次调用这个子程序时,Perl声明并初始化变量$n,而在接下来的调用中,这个声明表达式将被Perl忽略。每次子程序返回后,$n的当前值都会被保留下来,以备下次调用时再用。
state关键字的用法和my基本相似,但是不能在列表上下文中初始化数组类型的state变量。
use strict;
use feature 'state';
my @array = qw(a b c); # 正确
state @array = qw(a b c); # 错误
课后习题
(1)
use strict;
use feature 'state';sub total{my($sum) = 0;foreach (@_){$sum += $_;} $sum;
}my @fred = qw{1 3 5 7 9};
print &total(@fred);
(2)
use strict;
use feature 'state';sub total{my($sum) = 0;foreach (@_){$sum += $_;} $sum;
}my @fred = 1..1000;
print &total(@fred);
(3)
use strict;
use feature 'state';sub average{my $n = @_;my($sum) = 0;foreach (@_){$sum += $_;} $sum / $n;
}sub above_average{my $avg = &average(@_);foreach (@_){if($_ > $avg){print "$_\t";}}
}my @fred = 1..10;
print &above_average(@fred);
(4)
use strict;
use feature 'state';sub greet{my ($guest) = @_;state $last;if($last eq undef){print "Hi $guest! You are the first one here!\n";}else{print "Hi $guest! $last is also here!\n";}$last = $guest;
}&greet("Fred");
&greet("Barney");
(5)
use strict;
use feature 'state';sub greet{my ($guest) = @_;state @last;if(@last == 0){print "Hi $guest! You are the first one here!\n";}else{print "Hi $guest! I've seen: "."@last\t"."\n";}push @last, $guest;
}&greet("Fred");
&greet("Barney");
&greet("Wilma");
&greet("Betty");
参考文献
- Perl语言入门 第六版中文清晰PDF
这篇关于Perl语言入门读书笔记 | 四. 子程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!