18
module Test
  def self.model_method
    puts "this is a module method"
  end
end

class A
  include Test
end

A.model_method

これは次のエラーになります。

A:Class の未定義メソッド「model_method」(NoMethodError)

しかし、A. のメタクラスを使用すると動作します。

module Test
  def model_method
    puts "this is a module method"
  end
end

class A
  class << self
    include Test
  end
end

A.model_method

誰かがこれを説明できますか?

4

2 に答える 2

34

モジュールを含めるときにクラスメソッドとインスタンスメソッドの両方をクラスに混在させたい場合は、次のパターンに従うことができます。

module YourModule
  module ClassMethods
    def a_class_method
      puts "I'm a class method"
    end
  end

  def an_instance_method
    puts "I'm an instance method"
  end

  def self.included(base)
    base.extend ClassMethods
  end
end

class Whatever
  include YourModule
end

Whatever.a_class_method
# => I'm a class method

Whatever.new.an_instance_method
# => I'm an instance method

基本的にそれを過度に単純化するために、extendクラスメソッドをinclude追加し、インスタンスメソッドを追加します。モジュールが含まれると、その#includedメソッドが呼び出され、実際のクラスが含まれます。ここextendから、別のモジュールのいくつかのクラスメソッドを使用してクラスを作成できます。これは非常に一般的なパターンです。

参照:http ://api.rubyonrails.org/classes/ActiveSupport/Concern.html

于 2012-04-06T05:20:42.857 に答える
11

モジュールをインクルードすることは、そのインスタンス メソッドをコピーすることに似ています。

あなたの例では、コピーするインスタンス メソッドはありませんAmodel_methodは、実際にはTestのシングルトン クラスのインスタンス メソッドです。


与えられた:

module A
  def method
  end
end

これ:

module B
  include A
end

これに類似しています:

module B
  def method
  end
end

このように考えると、これは完全に理にかなっています。

module B
  class << self
    include A
  end
end

B.method

Bここでは、メソッドがモジュールのシングルトン クラスにコピーされ、B.

これは次のものとまったく同じであることに注意してください。

module B
  extend A
end

実際には、メソッドはコピーされていません。重複はありません。モジュールは、単にメソッド ルックアップ リストに含まれています。

于 2012-04-06T04:47:06.040 に答える