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.