其实就是所谓的函数。
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);
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”?)
在不加$号也不会产生歧义的情况下,函数调用时函数名前的$号是可以省略的。如下面这种情况,函数shift不仅可以省略$号,连参数列表两边的括号都可以省略:
use strict;
my @array = qw(a b c);
my @dealed = shift @array;
如果自己定义的子程序与Perl内置函数同名的话,为了调用自己定义的子程序,就必须加$号。所以更一般的情况是,调用自建函数时务必使用$号,除非你很确定Perl所有的内置函数名没有与你的自建函数同名。
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); # 错误
use strict;
use feature 'state';
sub total{
my($sum) = 0;
foreach (@_){
$sum += $_;
}
$sum;
}
my @fred = qw{1 3 5 7 9};
print &total(@fred);
use strict;
use feature 'state';
sub total{
my($sum) = 0;
foreach (@_){
$sum += $_;
}
$sum;
}
my @fred = 1..1000;
print &total(@fred);
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);
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");
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");