4

我有一个模块Top,它有模块AB. 在他们每个人中,我都想使用 module 的类方法C。为此,我必须包含C每个模块AB. 是否可以包含在内CTop以便所有子模块都可以访问其类方法?

例子:

# I'll extend module C in example to make it shorter

module C
  def foo; puts 'Foo!' end
end

module Top
  extend C

  module A  end
  module B  end
end

# That's how it works now

Top.foo
#=> "Foo!"
Top::A.foo
#=> NoMethodError: undefined method `foo' for Top::A:Module
Top::B.foo
#=> NoMethodError: undefined method `foo' for Top::B:Module

# That's how I want it to work

Top.foo
#=> "Foo!"
Top::A.foo
#=> "Foo!"
Top::B.foo
#=> "Foo!"
4

2 回答 2

3

好吧,实际上它可能是
OP 更新的代码,所以这是我的实现:

class Module
  def submodules
    constants.collect {|const_name| const_get(const_name)}.select {|const| const.class == Module}
  end
end


module C
  # this gets called when the module extends another class or module
  # luckily it does _not_ get called when we extend via :send
  def self.extended(base)
    # collect all submodules and extend them with self
    base.submodules.each{|m| m.send :extend, self }
  end
  def c1
    puts "c1"
  end
end

module Top
  module A;end
  module B;end
  # extend needs to go at the end - otherwise Top doesnt know about its submodules
  extend C
end

Top.c1     # => "c1"
Top::A.c1  # => "c1"
Top::B.c1  # => "c1"
于 2012-12-19T09:00:06.223 回答
0

没有内置的方式。你必须像这样实现它:

class Module
  def extend_each_module m
    constants.each do |sym|
      const_get(sym).instance_eval{extend(m) if kind_of?(Module)}
    end
  end
end

module C
  def foo
    puts "Foo!"
  end
end

module Top
  module A; end
  module B; end
  extend C
  extend_each_module C
end

Top.foo
# => Foo!
Top::A.foo
# => Foo!
Top::B.foo
# => Foo!
于 2012-12-19T08:51:26.867 回答