Safari Books Online is a digital library providing on-demand subscription access to thousands of learning resources.
That subroutine called larger_of_fred_or_barney would be much more
useful if it didn’t force you to use the global variables $fred and $barney. If you wanted to get the larger value
from $wilma and $betty, you currently have to copy those into
$fred and $barney before you can use larger_of_fred_or_barney. And if you had
something useful in those variables, you’d have to first copy those to
other variables, say $save_fred and
$save_barney. And then, when you’re
done with the subroutine, you’d have to copy those back to $fred and $barney again.
Luckily, Perl has subroutine arguments. To pass an argument list to the subroutine, simply place the list expression, in parentheses, after the subroutine invocation, like this:
$n = &max(10, 15); # This sub call has two parameters
Perl passes the list to the subroutine; that
is, Perl makes the list available for the subroutine to use however it
needs to. Of course, you have to store this list somewhere, so Perl
automatically stores the parameter list (another name for the argument
list) in the special array variable named @_ for the duration of the subroutine. You can
access this array to determine both the number of arguments and the
value of those arguments.
This means that the first subroutine parameter is in $_[0], the second one is stored in $_[1], and so on. But—and here’s an important
note—these variables have nothing whatsoever to do with the $_ variable, any more than $dino[3] (an element of the @dino array) has to do with $dino (a completely distinct scalar variable).
It’s just that the parameter list must be in some array variable for
your subroutine to use it, and Perl uses the array @_ for this purpose.
Now, you could write the subroutine &max to look a little like the subroutine
&larger_of_fred_or_barney, but
instead of using $fred you
could use the first subroutine parameter ($_[0]), and instead of using $barney, you could use
the second subroutine parameter
($_[1]). And so you
could end up with something like this:
sub max {
# Compare this to &larger_of_fred_or_barney
if ($_[0] > $_[1]) {
$_[0];
} else {
$_[1];
}
}
Well, as we said, you could do that. But it’s pretty ugly with all of those subscripts, and hard to read, write, check, and debug, too. You’ll see a better way in a moment.
There’s another problem with this subroutine. The name &max is nice and short, but it doesn’t
remind us that this subroutine works properly only if called with
exactly two parameters:
$n = &max(10, 15, 27); # Oops!
max ignores the extra
parameters since it never looks at $_[2]. Perl doesn’t care whether there’s
something in there or not. Perl doesn’t care about insufficient
parameters either—you simply get
undef if you look beyond the end of
the @_ array, as with any other
array. You’ll see how to make a better &max, which works with any number of
parameters, later in this chapter.
The @_ variable is private to
the subroutine;[103] if there’s a global value in @_, Perl saves it before it invokes the next
subroutine and restores its previous value upon return from that
subroutine.[104] This also means that a subroutine can pass arguments to
another subroutine without fear of losing its own @_ variable—the nested subroutine invocation
gets its own @_ in the same way. Even
if the subroutine calls itself recursively, each invocation gets a new
@_, so @_ is always the parameter list for the
current subroutine invocation.
[103] Unless there’s an ampersand in front of the name for the
invocation, and no parentheses (or arguments) afterward, in which
case the @_ array is inherited
from the caller’s context. That’s generally a bad idea, but is
occasionally useful.