Bug #11808 open
Different behavior between Enumerable#grep and Array#grep
Added by BenOlive (Ben Olive) over 9 years ago.
Updated about 1 year ago.
Description
Regex special global variables are available within the block for Array#grep
, but are nil within the block for Enumerable#grep
.
Here is an example that explains it better:
class Test
include Enumerable
def each
return enum_for ( :each ) unless block_given?
yield "Hello"
yield "World"
end
end
enum = Test . new
array = [ "Hello" , "World" ]
enum . grep ( /^(.)/ ) { $1 } # => [nil, nil]
array . grep ( /^(.)/ ) { $1 } # => ["H", "W"]
Tested on 2.0.0, 2.1.5, & 2.2.2
Related issues
1 (1 open — 0 closed )
Array#grep
is actually Enumerable#grep
:
Array . instance_method ( :grep ). owner
=> Enumerable
If I had to guess, the cause of the difference is that Array#each
is implemented in C, and Test#each
is implemented in Ruby, and this affects Regexp special variable scope. You see similar behavior as Array in other classes that implement #each
in C, such as Range or File.
The documentation for the special global variables states: These global variables are thread-local and method-local variables.
This indicates to me that the bug is that the variables are accessible inside the Array#each
block, since that block executes inside the current method, it's not local to the Array#each
method. However, I would assume removing the current behavior would break too much existing code.
Tracker changed from Bug to Feature
Description updated (diff )
ruby -v deleted (2.2.2 )
Backport deleted (2.0.0: UNKNOWN, 2.1: UNKNOWN, 2.2: UNKNOWN )
Subject changed from DIfferent behavior between Enumerable#grep and Array#grep to Different behavior between Enumerable#grep and Array#grep
Tracker changed from Feature to Bug
Backport set to 2.5: UNKNOWN, 2.6: UNKNOWN
It is a bug. It has been hidden for 10+ years and seems to be very difficult to fix.
It should be fixed in the long run.
Matz.
Assignee set to ko1 (Koichi Sasada)
Sorry we need more time to consider.
This appears to be fixed in 3.3.0dev as of Nov 2023,
irb ( main ): 001 * class Test
irb ( main ): 002 * include Enumerable
irb ( main ): 003 * def each
irb ( main ): 004 * return enum_for ( :each ) unless block_given?
irb ( main ): 005 * yield "Hello"
irb ( main ): 006 * yield "World"
irb ( main ): 007 * end
irb ( main ): 00 8 > end
=> :each
irb ( main ): 00 9 >
irb ( main ): 010 > enum = Test . new
=> #<Test:0x0000000102ba5038>
irb ( main ): 011 > array = [ "Hello" , "World" ]
=> [ "Hello" , "World" ]
irb ( main ): 012 >
irb ( main ): 013 > enum . grep ( /^(.)/ ) { $1 }
=> [ "H" , "W" ]
However, it was still broken in 3.2.2. It's not clear to me when the behavior changed.
Status changed from Open to Assigned
Related to Bug #20807 : String#gsub fails when called from string subclass with a block passed added
Also available in: Atom
PDF
Like 0
Like 0 Like 0 Like 0 Like 0 Like 0 Like 0 Like 0 Like 0 Like 0