Class: Enumerator
- Inherits:
-
Object
- Object
- Enumerator
- Includes:
- Enumerable
- Defined in:
- mrbgems/mruby-enumerator/mrblib/enumerator.rb,
mrbgems/mruby-enum-lazy/mrblib/lazy.rb,
mrbgems/mruby-enum-chain/mrblib/chain.rb
Overview
A class which allows both internal and external iteration.
An Enumerator can be created by the following methods.
Most methods have two forms: a block form where the contents are evaluated for each item in the enumeration, and a non-block form which returns a new Enumerator wrapping the iteration.
enumerator = %w(one two three).each
puts enumerator.class # => Enumerator
enumerator.each_with_object("foo") do |item, obj|
puts "#{obj}: #{item}"
end
# foo: one
# foo: two
# foo: three
enum_with_obj = enumerator.each_with_object("foo")
puts enum_with_obj.class # => Enumerator
enum_with_obj.each do |item, obj|
puts "#{obj}: #{item}"
end
# foo: one
# foo: two
# foo: three
This allows you to chain Enumerators together. For example, you can map a list's elements to strings containing the index and the element as a string via:
puts %w[foo bar baz].map.with_index { |w, i| "#{i}:#{w}" }
# => ["0:foo", "1:bar", "2:baz"]
An Enumerator can also be used as an external iterator. For example, Enumerator#next returns the next value of the iterator or raises StopIteration if the Enumerator is at the end.
e = [1,2,3].each # returns an enumerator object.
puts e.next # => 1
puts e.next # => 2
puts e.next # => 3
puts e.next # raises StopIteration
You can use this to implement an internal iterator as follows:
def ext_each(e)
while true
begin
vs = e.next_values
rescue StopIteration
return $!.result
end
y = yield(*vs)
e.feed y
end
end
o = Object.new
def o.each
puts yield
puts yield(1)
puts yield(1, 2)
3
end
# use o.each as an internal iterator directly.
puts o.each {|*x| puts x; [:b, *x] }
# => [], [:b], [1], [:b, 1], [1, 2], [:b, 1, 2], 3
# convert o.each to an external iterator for
# implementing an internal iterator.
puts ext_each(o.to_enum) {|*x| puts x; [:b, *x] }
# => [], [:b], [1], [:b, 1], [1, 2], [:b, 1, 2], 3
Direct Known Subclasses
Defined Under Namespace
Classes: Chain, Generator, Lazy, Yielder
Constant Summary
Constants included from Enumerable
Instance Method Summary collapse
- #+(other) ⇒ Object
-
#each(*argv, &block) ⇒ Object
call-seq: enum.each { |elm| block } -> obj enum.each -> enum enum.each(*appending_args) { |elm| block } -> obj enum.each(*appending_args) -> an_enumerator.
-
#each_with_index(&block) ⇒ Object
call-seq: e.each_with_index {|(*args), idx| ... } e.each_with_index.
-
#feed(value) ⇒ Object
call-seq: e.feed obj -> nil.
-
#initialize(obj, method = :each, *args, **kwd) ⇒ Enumerator
constructor
Creates a new Enumerator object, which can be used as an Enumerable.
- #initialize_copy(obj) ⇒ Object
- #inspect ⇒ Object
-
#next ⇒ Object
call-seq: e.next -> object.
-
#next_values ⇒ Object
call-seq: e.next_values -> array.
-
#peek ⇒ Object
call-seq: e.peek -> object.
-
#peek_values ⇒ Object
call-seq: e.peek_values -> array.
-
#rewind ⇒ Object
call-seq: e.rewind -> e.
- #size ⇒ Object
-
#with_index(offset = 0, &block) ⇒ Object
call-seq: e.with_index(offset = 0) {|(*args), idx| ... } e.with_index(offset = 0).
-
#with_object(object, &block) ⇒ Object
call-seq: e.each_with_object(obj) {|(*args), obj| ... } e.each_with_object(obj) e.with_object(obj) {|(*args), obj| ... } e.with_object(obj).
Methods included from Enumerable
#all?, #any?, #chain, #chunk, #chunk_while, #collect, #count, #cycle, #detect, #drop, #drop_while, #each_cons, #each_entry, #each_slice, #each_with_object, #entries, #filter_map, #find_all, #find_index, #first, #flat_map, #grep, #grep_v, #group_by, #hash, #include?, #inject, #lazy, #max, #max_by, #min, #min_by, #minmax, #minmax_by, #none?, #one?, #partition, #reject, #reverse_each, #sort, #sort_by, #sum, #take, #take_while, #tally, #to_h, #uniq, #zip
Constructor Details
#initialize(obj, method = :each, *args, **kwd) ⇒ Enumerator
Creates a new Enumerator object, which can be used as an Enumerable.
In the first form, iteration is defined by the given block, in
which a "yielder" object, given as block parameter, can be used to
yield a value by calling the yield method (aliased as <<):
fib = Enumerator.new do |y|
a = b = 1
loop do
y << a
a, b = b, a + b
end
end
p fib.take(10) # => [1, 1, 2, 3, 5, 8, 13, 21, 34, 55]
In the second, deprecated, form, a generated Enumerator iterates over the given object using the given method with the given arguments passed. This form is left only for internal use.
Use of this form is discouraged. Use Kernel#enum_for or Kernel#to_enum instead.
117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 117 def initialize(obj=NONE, meth=:each, *args, **kwd, &block) if block obj = Generator.new(&block) elsif NONE.equal?(obj) raise ArgumentError, "wrong number of arguments (given 0, expected 1+)" end @obj = obj @meth = meth @args = args @kwd = kwd @fib = nil @dst = nil @lookahead = nil @feedvalue = nil @stop_exc = false end |
Instance Method Details
#+(other) ⇒ Object
12 13 14 |
# File 'mrbgems/mruby-enum-chain/mrblib/chain.rb', line 12 def +(other) Chain.new(self, other) end |
#each(*argv, &block) ⇒ Object
call-seq: enum.each { |elm| block } -> obj enum.each -> enum enum.each(*appending_args) { |elm| block } -> obj enum.each(*appending_args) -> an_enumerator
Iterates over the block according to how this Enumerator was constructed. If no block and no arguments are given, returns self.
Examples
Array.new(3) #=> [nil, nil, nil]
Array.new(3) { |i| i } #=> [0, 1, 2]
Array.to_enum(:new, 3).to_a #=> [0, 1, 2]
Array.to_enum(:new).each(3).to_a #=> [0, 1, 2]
obj = Object.new
def obj.each_arg(a, b=:b, *rest)
yield a
yield b
yield rest
:method_returned
end
enum = obj.to_enum :each_arg, :a, :x
enum.each.to_a #=> [:a, :x, []]
enum.each.equal?(enum) #=> true
enum.each { |elm| elm } #=> :method_returned
enum.each(:y, :z).to_a #=> [:a, :x, [:y, :z]]
enum.each(:y, :z).equal?(enum) #=> false
enum.each(:y, :z) { |elm| elm } #=> :method_returned
285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 285 def each(*argv, &block) obj = self if 0 < argv.length obj = self.dup args = obj.instance_eval{@args} if !args.empty? args = args.dup args.concat argv else args = argv.dup end obj.instance_eval{@args = args} end return obj unless block __enumerator_block_call(&block) end |
#each_with_index(&block) ⇒ Object
call-seq: e.each_with_index {|(*args), idx| ... } e.each_with_index
Same as Enumerator#with_index(0), i.e. there is no starting offset.
If no block is given, a new Enumerator is returned that includes the index.
191 192 193 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 191 def each_with_index(&block) with_index(0, &block) end |
#feed(value) ⇒ Object
call-seq: e.feed obj -> nil
Sets the value to be returned by the next yield inside e.
If the value is not set, the yield returns nil.
This value is cleared after being yielded.
# Array#map passes the array's elements to "yield" and collects the
# results of "yield" as an array.
# Following example shows that "next" returns the passed elements and
# values passed to "feed" are collected as an array which can be
# obtained by StopIteration#result.
e = [1,2,3].map
p e.next #=> 1
e.feed "a"
p e.next #=> 2
e.feed "b"
p e.next #=> 3
e.feed "c"
begin
e.next
rescue StopIteration
p $!.result #=> ["a", "b", "c"]
end
o = Object.new
def o.each
x = yield # (2) blocks
p x # (5) => "foo"
x = yield # (6) blocks
p x # (8) => nil
x = yield # (9) blocks
p x # not reached w/o another e.next
end
e = o.to_enum
e.next # (1)
e.feed "foo" # (3)
e.next # (4)
e.next # (7)
# (10)
535 536 537 538 539 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 535 def feed(value) raise TypeError, "feed value already set" if @feedvalue @feedvalue = value nil end |
#initialize_copy(obj) ⇒ Object
135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 135 def initialize_copy(obj) raise TypeError, "can't copy type #{obj.class}" unless obj.kind_of? Enumerator raise TypeError, "can't copy execution context" if obj.instance_eval{@fib} meth = args = kwd = fib = nil obj.instance_eval { obj = @obj meth = @meth args = @args kwd = @kwd } @obj = obj @meth = meth @args = args @kwd = kwd @fib = nil @lookahead = nil @feedvalue = nil self end |
#inspect ⇒ Object
232 233 234 235 236 237 238 239 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 232 def inspect if @args && @args.size > 0 args = @args.join(", ") "#<#{self.class}: #{@obj.inspect}:#{@meth}(#{args})>" else "#<#{self.class}: #{@obj.inspect}:#{@meth}>" end end |
#next ⇒ Object
call-seq: e.next -> object
Returns the next object in the enumerator, and move the internal position forward. When the position reached at the end, StopIteration is raised.
Example
a = [1,2,3]
e = a.to_enum
p e.next #=> 1
p e.next #=> 2
p e.next #=> 3
p e.next #raises StopIteration
Note that enumeration sequence by next does not affect other non-external
enumeration methods, unless the underlying iteration methods itself has
side-effect
327 328 329 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 327 def next next_values.__svalue end |
#next_values ⇒ Object
call-seq: e.next_values -> array
Returns the next object as an array in the enumerator, and move the internal position forward. When the position reached at the end, StopIteration is raised.
This method can be used to distinguish yield and yield
nil.
Example
o = Object.new
def o.each
yield
yield 1
yield 1, 2
yield nil
yield [1, 2]
end
e = o.to_enum
p e.next_values
p e.next_values
p e.next_values
p e.next_values
p e.next_values
e = o.to_enum
p e.next
p e.next
p e.next
p e.next
p e.next
## yield args next_values next
# yield [] nil
# yield 1 [1] 1
# yield 1, 2 [1, 2] [1, 2]
# yield nil [nil] nil
# yield [1, 2] [[1, 2]] [1, 2]
Note that next_values does not affect other non-external enumeration
methods unless underlying iteration method itself has side-effect
375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 375 def next_values if @lookahead vs = @lookahead @lookahead = nil return vs end raise @stop_exc if @stop_exc curr = Fiber.current if !@fib || !@fib.alive? @dst = curr @fib = Fiber.new do result = each do |*args| feedvalue = nil Fiber.yield args if @feedvalue feedvalue = @feedvalue @feedvalue = nil end feedvalue end @stop_exc = StopIteration.new "iteration reached an end" @stop_exc.result = result Fiber.yield nil end @lookahead = nil end vs = @fib.resume curr if @stop_exc @fib = nil @dst = nil @lookahead = nil @feedvalue = nil raise @stop_exc end vs end |
#peek ⇒ Object
call-seq: e.peek -> object
Returns the next object in the enumerator, but doesn't move the internal position forward. If the position is already at the end, StopIteration is raised.
Example
a = [1,2,3]
e = a.to_enum
p e.next #=> 1
p e.peek #=> 2
p e.peek #=> 2
p e.peek #=> 2
p e.next #=> 2
p e.next #=> 3
p e.next #raises StopIteration
435 436 437 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 435 def peek peek_values.__svalue end |
#peek_values ⇒ Object
call-seq: e.peek_values -> array
Returns the next object as an array, similar to Enumerator#next_values, but doesn't move the internal position forward. If the position is already at the end, StopIteration is raised.
Example
o = Object.new
def o.each
yield
yield 1
yield 1, 2
end
e = o.to_enum
p e.peek_values #=> []
e.next
p e.peek_values #=> [1]
p e.peek_values #=> [1]
e.next
p e.peek_values #=> [1, 2]
e.next
p e.peek_values # raises StopIteration
465 466 467 468 469 470 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 465 def peek_values if @lookahead.nil? @lookahead = next_values end @lookahead.dup end |
#rewind ⇒ Object
call-seq: e.rewind -> e
Rewinds the enumeration sequence to the beginning.
If the enclosed object responds to a "rewind" method, it is called.
480 481 482 483 484 485 486 487 488 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 480 def rewind @obj.rewind if @obj.respond_to? :rewind @fib = nil @dst = nil @lookahead = nil @feedvalue = nil @stop_exc = false self end |
#size ⇒ Object
241 242 243 244 245 246 247 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 241 def size if @size @size elsif @obj.respond_to?(:size) @obj.size end end |
#with_index(offset = 0, &block) ⇒ Object
call-seq: e.with_index(offset = 0) {|(*args), idx| ... } e.with_index(offset = 0)
Iterates the given block for each element with an index, which
starts from offset. If no block is given, returns a new Enumerator
that includes the index, starting from offset
offsetthe starting index to use
166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 166 def with_index(offset=0, &block) return to_enum :with_index, offset unless block if offset.nil? offset = 0 else offset = offset.__to_int end n = offset - 1 __enumerator_block_call do |*i| n += 1 block.call i.__svalue, n end end |
#with_object(object, &block) ⇒ Object
call-seq: e.each_with_object(obj) {|(*args), obj| ... } e.each_with_object(obj) e.with_object(obj) {|(*args), obj| ... } e.with_object(obj)
Iterates the given block for each element with an arbitrary object, obj,
and returns obj
If no block is given, returns a new Enumerator.
223 224 225 226 227 228 229 230 |
# File 'mrbgems/mruby-enumerator/mrblib/enumerator.rb', line 223 def with_object(object, &block) return to_enum(:with_object, object) unless block __enumerator_block_call do |i| block.call [i,object] end object end |