What if you want to jump out of the block that contains the innermost block, or to put it another way, exit from two nested blocks at once? In C, you'd resort to that much maligned
goto
to get you out. No such
kludge is required in Perl; you can use
last
,
next
, and
redo
on any enclosing block by giving the block a name with a
label
.
A label is yet another type of name from yet another namespace following the same rules as scalars, arrays, hashes, and subroutines. As we'll see, however, a label doesn't have a special prefix punctuation character (like
$
for scalars,
&
for subroutines, and so on), so a label named
print
conflicts with the reserved word
print
and would not be allowed. For this reason,
you should choose labels that consist entirely of uppercase letters and digits, which will never be chosen for a reserved word in the future. Besides, using all uppercase stands out better within the text of a mostly lowercase program.
Once you've chosen your label, place it immediately in front of the statement containing the block followed by a
colon, like this:
SOMELABEL: while (
condition
) {
statement
;
statement
;
statement
;
if (
nuthercondition
) {
last SOMELABEL;
}
}
We added
SOMELABEL
as a parameter to
last
. This tells Perl to exit the block named
SOMELABEL
, rather than just the innermost block. In this case, we don't have anything but the innermost block. But suppose we had
nested loops:
OUTER:
for ($i = 1; $i <= 10; $i++) {
INNER: for ($j = 1; $j <= 10; $j++) {
if ($i * $j == 63) {
print "$i times $j is 63!\n";
last OUTER;
}
if ($j >= $i) {
next OUTER;
}
}
}
This set of statements tries all successive values of two small numbers multiplied together until it finds a pair whose
product is 63 (7 and 9). Once the pair is found, there's no point in testing other numbers, so the first
if
statement exits both
for
loops using
last
with a label. The second
if
ensures that the bigger of the two numbers will always be the first one by skipping to the next iteration of the outer loop as soon as the condition would no longer hold. This means that the numbers will be tested with (
$i
,
$j
) being (
1
,
1
), (
2
,
1
), (
2
,
2
), (
3
,
1
), (
3
,
2
), (
3
,
3
), (
4
,
1
), and so on.
Even if the innermost block is labeled, the
last
,
next
, and
redo
statements without the optional parameter (the label) still operate with respect to that innermost block. Also, you can't use labels to jump into a block - just out of a block. The
last
,
next
, or
redo
has to be within the block.