那么has_many可以給我們帶來什么呢?類方法has_many在被執行的時候,給Topic的對象實例添加了一系列方法:posts, posts, orders.push......等等。所以當我們在model里面聲明has_many,belongs_to等對象關系的時候,一系列相關的對象方法就被自動添加進來了。 讓我們來自己試試看吧:
module M
def self.included(c)
c.extend(G)
end
module G
def generate_method(*args)
args.each do |method_name|
define_method(method_name) { puts method_name }
end
end
end
end
class C
include M
generate_method :method1, :method2
end
c = C.new
c.method1
c.method2
我們定義了一個聲明generate_method,可以接受多個symbol,來動態的創建同名的方法。現在我們在類C里面使用這個聲明:generate_method :method1, :method2,當然我們需要include模塊M。為什么ActiveRecord的model不需要include相關的模塊呢?當然是因為Topic的父類ActiveRecord::Base已經include了模塊Associations了。
類C通過include模塊M,調用了模塊M的一個included回調接口,讓類C去extend模塊G,換句話來說就是,通過include模塊M,來給類C動態添加一個類方法generate_method。
通過在類C里面聲明generate_method :method1, :method2,讓類C動態的添加了兩個實例方法method1,method2,是不是很有意思? 實際上rails的對象關聯聲明也是以同樣的方式實現的。