|
子例程,函数
子例程,函数的作用域是指在程序中能够看到它的位置范围。子例程是全局的,可以放在脚本中的任意位置,甚至放在其他脚本文件中。当使用来自其他文件的子例程时,应当使用关键字do,require或use将它加载到脚本里。若要调用一个子例程,用户可以在子例程前加上&符号,或在子例程前加上do关键字,也可以在子例程名后面加上一组括号。如果使用了向前引用(forward reference),在调用子例程时就不需要提供&或括号。
1 Subroutine declaration:
2 sub subroutine_name;
3 Subroutine definition:
4 sub subroutine_name{ Block }
5 Subroutine call:
6 do subroutine_name;
7 &subroutine_name;
8 subroutine_name();
9 subroutine_name;
10 Subroutine call with parameters:
11 &subroutine_name( parameter1,parameter2,.......);
12 subroutine_name( parameter1,parameter2,.......);
示例代码:
1 #!/user/bin/perl
2
3 sub greetme
4 {
5 print "Welcome,Valkommen till,Bienvenue!\n";
6 }
7
8 &greetme if defined &greetme; #使用defined函数检查子例程是否已经定义。在使用子例程名时,要求必须提供&标记。
9
10 print "Program continues ....\n";
11
12 &greetme;
13
14 bye();
15 &bye;
16 #bye; 此种方式调用需使用向前引用,类似C程序中的声明。
17 sub bye;
18 bye;
19
20 sub bye
21 {
22 print "Bye,adjo,adieu.\n";
23 }
24
25 bye();
26 &bye;
27 bye;
运行结果:
1 Welcome,Valkommen till,Bienvenue!
2 Program continues ....
3 Welcome,Valkommen till,Bienvenue!
4 Bye,adjo,adieu.
5 Bye,adjo,adieu.
6 Bye,adjo,adieu.
7 Bye,adjo,adieu.
8 Bye,adjo,adieu.
9 Bye,adjo,adieu.
参数传递
函数把接收到的参数保存到特殊的perl数组@_中,其中每个元素($_[0],$_[1],.....)称作一个参数。不论参数是标量型还是数组型的,用户把参数传给子例程时,perl默认按引用的方式调用它们。用户可以通过改变@_数组中的值来改变相应实际参数的值。
示例代码:
1 #!/usr/bin/perl
2
3 sub params
4 {
5 print 'The values in the @_array are ',"@_\n";#将打印整个数组
6 print "The first value is $_[0]\n";
7 print "The last value is ",pop(@_),"\n";#最后一个参数被弹出@_数组,实际参数的个数并未改变
8
9 foreach $value (@_)
10 {
11 $value+=5;
12 print "The value is $value","\n";
13 }
14 }
15
16 @n=(1,2,3,4,5);
17 params(@n);
18 print "Black in main\n";
19 print "The new values are @n \n";
运行结果:
1 The values in the @_array are 1 2 3 4 5
2 The first value is 1
3 The last value is 5
4 The value is 6
5 The value is 7
6 The value is 8
7 The value is 9
8 Black in main
9 The new values are 6 7 8 9 5 #前四个参数值已经改变,最后一个参数值没有发生变化,因为子例程中执行了pop(@_).
local与my区别
任何使用local函数声明的变量都是动态拷贝的。该变量不但在创建它的块中可见,而且对于任何从该代码块调用的函数中也是可见的。与local函数声明的变量不同,my函数声明得到的所有变量都只在声明它的子例程中可见,对于该子例程调用的任何其他子例程都是不可见的
内部->外部:
1)my和local都只在一个block里有效,出去就失效;
2)但是local的变量可以继续在这个block中调用的子程序中存在;
3)如果有与外界同名的变量,两者在block退出后都不影响外界同名变量;
外部->内部:
如果在一个block中有一个my修饰的变量和外界的一个变量同名,而且又需要在这个block中使用外界变量时,两个办法:第一个办法,用main的package修饰这个变量名,$main::global。第二个办法,用our修饰,our $global,那么该block中接下来出现的所有$global都是外界的global
示例代码:
1 #!/usr/bin/perl
2 $x = 9;
3 sub mm{
4 print "$x\n";
5 }
6 sub gg_my{
7 my $x = 11;
8 print "call-my $x\n";
9 our $x;
10 print "our-global $x\n";
11 mm;
12 }
13 sub gg_local {
14 local $x = 20;
15 print "call-local $x\n";
16 mm;
17 }
18 gg_my;
19 gg_local;
20 print $x;
运行结果:
1 call-my 11
2 our-global 9
3 9
4 call-local 20
5 20
6 9
创建指针和访问指针地址
赋值 | 创建引用 | 访问地址 | 使用箭头运算符访问地址 | $sca = 5 | $p=\$sca | print $$p | | @arr=(4,5,6) | $p=\@arr | print @$p | $p->[0] | | | print $$p[0] | | %hash=(key=>'value') | $p=\%hash | print %$p | $p->{key} | | | print $$p{key} | |
指针参数
当一个子例程接收参数时,便会把参数保存到@_数组中。当把两个数组同时传递给一个子例程时,这两个数组都将保存到@_数组中,并连接成单个列表。如果不知道其中前一个数组长度的话,基本上就不可能再拆开这两个数组了。不过,如果传递给子例程的是两个指针,分别含有这两个数组地址的话,就可以方便地实现数组内容的分离。
示例代码:
1 #!/usr/bin/perl
2
3 my @list1=(1..100);
4 my @list2=(5,10,15,20);
5
6 print "the total is :" ,&addemup(\@list1,\@list2),".\n";
7
8 sub addemup
9 {
10 my( $arr1,$arr2) =( shift,shift);# The two pointers are shifted from @_
11
12 my $total=0;
13 print $arr1,"\n";
14 print $arr2,"\n";
15
16 foreach $num(@$arr1,@$arr2)
17 {
18 $total+=$num;
19 }
20 $total; #相当于 return $total;
21 }
执行结果:
1 ARRAY(0x182a944)
2 ARRAY(0x183728c)
3 the total is :5100.
文件句柄的引用
示例代码:
1 #!/usr/bin/perl
2
3 open( README,"D:\\perlwork1\\perl.pl") || die;
4
5 &readit(\*README);#reference to a typeglob
6
7 sub readit
8 {
9 my ($passwd)=@_;
10 print "\$passwd is a $passwd.\n";
11 while(<$passwd>)
12 {
13 print;
14 }
15 }
执行结果:
1 $passwd is a GLOB(0x182a964).
2 #!usr/bin/perl
3 。。。。
|
|