In Control flow§
See primary documentation in context for when
The when
block is similar to an if
block and either or both can be used in an outer block; they also both have a "statement modifier" form. But there is a difference in how following code in the same, outer block is handled: When the when
block is executed, control is passed to the enclosing block and following statements are ignored; but when the if
block is executed, following statements are executed. [1]
The following examples should illustrate the if
or when
block's default behavior assuming no special exit or other side effect statements are included in the if
or when
blocks:
{ if X {...} # if X is true in Boolean context, block is executed # following statements are executed regardless } { when X {...} # if X is true in Boolean context, block is executed # and control passes to the outer block # following statements are NOT executed }
Should the if
and when
blocks above appear at file scope, following statements would be executed in each case.
There is one other feature when
has that if
doesn't: the when
's Boolean context test defaults to $_ ~~
while the if
's does not. That has an effect on how one uses the X in the when
block without a value for $_
(it's Any
in that case and Any
smartmatches on True
: Any ~~ True
yields True
). Consider the following:
{ my $a = 1; my $b = True; when $a { say 'a' }; # no output when so $a { say 'a' } # a ("so $a" 'so' coerces $a to Boolean context True # which matches with Any) when $b { say 'b' }; # no output (this statement won't be run) }
Finally, when
's statement modifier form does not affect execution of following statements either inside or outside of another block:
say "foo" when X; # if X is true statement is executed # following statements are not affected
Since a successful match will exit the block, the behavior of this piece of code:
$_ = True; my $a; { $a = do when .so { "foo" } }; say $a; # OUTPUT: «(Any)»
is explained since the do
block is abandoned before any value is stored or processed. However, in this case:
$_ = False; my $a; { $a = do when .so { "foo" } }; say $a; # OUTPUT: «False»
the block is not abandoned since the comparison is false, so $a
will actually get a value.