Sha256: a7fe0099d02a42d70db848d15504b1cddb6b8fa1ca26315141ad01a2ea737cd5

Contents?: true

Size: 1.98 KB

Versions: 176

Compression:

Stored size: 1.98 KB

Contents

module Tins
  # This class can create generator objects, that can produce all tuples, that
  # would be created by as many for-loops as dimensions were given.
  #
  # The generator
  #  g = Tins::Generator[1..2, %w[a b c]]
  # produces
  #  g.to_a # => [[1, "a"], [1, "b"], [1, "c"], [2, "a"], [2, "b"], [2, "c"]]
  #
  # The 'each' method can be used to iterate over the tuples
  #  g.each { |a, b| puts "#{a} #{b}" }
  # and Tins::Generator includes the Enumerable module, so
  # Enumerable.instance_methods can be used as well:
  #  g.select { |a, b| %w[a c].include? b  } # => [[1, "a"], [1, "c"], [2, "a"], [2, "c"]]
  #
  class Generator
    include Enumerable

    # Create a new Generator object from the enumberables _enums_.
    def self.[](*enums)
      new(enums)
    end

    # Create a new Generator instance. Use the objects in the Array _enums_
    # as dimensions. The should all respond to the :each method (see module
    # Enumerable in the core ruby library).
    def initialize(enums)
      @enums, @iterators, @n = [], [], 0
      enums.each { |e| add_dimension(e) }
    end

    # Iterate over all tuples produced by this generator and yield to them.
    def each(&block) # :yield: tuple
      recurse(&block)
      self
    end

    def recurse(tuple = [ nil ] * @n, i = 0, &block)
      if i < @n - 1 then
        @enums[i].__send__(@iterators[i]) do |x|
          tuple[i] = x
          recurse(tuple, i + 1, &block)
        end
      else
        @enums[i].__send__(@iterators[i]) do |x|
          tuple[i] = x
          yield tuple.dup
        end
      end
    end
    private :recurse

    # Add another dimension to this generator. _enum_ is an object, that ought
    # to respond to the _iterator_ method (defaults to :each).
    def add_dimension(enum, iterator = :each)
      @enums << enum
      @iterators << iterator
      @n += 1
    end

    # Return the size of this generator, that is the number of its dimensions.
    def size
      @enums.size
    end
  end
end

require 'tins/alias'

Version data entries

176 entries across 158 versions & 9 rubygems

Version Path
tins-1.38.0 lib/tins/generator.rb
tins-1.37.1 lib/tins/generator.rb
tins-1.37.0 lib/tins/generator.rb
tins-1.36.1 lib/tins/generator.rb
tins-1.36.0 lib/tins/generator.rb
tins-1.35.0 lib/tins/generator.rb
tins-1.34.0 lib/tins/generator.rb
tins-1.33.0 lib/tins/generator.rb
tdiary-5.2.4 vendor/bundle/ruby/3.1.0/gems/tins-1.31.1/lib/tins/generator.rb
tins-1.32.1 lib/tins/generator.rb
tins-1.32.0 lib/tins/generator.rb
tdiary-5.2.3 vendor/bundle/ruby/3.1.0/gems/tins-1.31.1/lib/tins/generator.rb
tdiary-5.2.2 vendor/bundle/ruby/3.1.0/gems/tins-1.31.1/lib/tins/generator.rb
tins-1.31.1 lib/tins/generator.rb
tdiary-5.2.1 vendor/bundle/ruby/3.1.0/gems/tins-1.31.0/lib/tins/generator.rb
tins-1.31.0 lib/tins/generator.rb
tins-1.30.0 lib/tins/generator.rb
tdiary-5.2.0 vendor/bundle/ruby/3.0.0/gems/tins-1.29.1/lib/tins/generator.rb
tdiary-5.2.0 vendor/bundle/ruby/2.7.0/gems/tins-1.29.1/lib/tins/generator.rb
tdiary-5.1.7 vendor/bundle/ruby/3.0.0/gems/tins-1.29.1/lib/tins/generator.rb