There are two different semantics with block arguments:
- Inline blocks (inlined with
yield
) - Captured blocks (called with
block.call
or passed as a proc)
In your example, the block is captured because you pass it on to sort_by
. But captured blocks requires restrictions for input and output types (see Capturing blocks).
That makes the second example work.
An alternative would be to use an inline block which is forwarded not by passing a captured proc, but forwarding with yield:
def forward_block(&)
[1, 2, 3].sort_by { |a| yield a }
end
forward_block { |a| -a }
The error message is probably not very helpful in this case, and we need to improve on that.
I think, with the new unnamed block argument syntax, we could move forward to disallow using named block arguments with yield, which would help separate both variants more clearly. Then a named block argument would always require type restrictions, which would make this example error in def forward_block(&block)
.