那么has_many可以給我們帶來什么呢?類方法has_many在被執(zhí)行的時候,給Topic的對象實例添加了一系列方法:posts, posts, orders.push......等等。所以當(dāng)我們在model里面聲明has_many,belongs_to等對象關(guān)系的時候,一系列相關(guān)的對象方法就被自動添加進(jìn)來了。 讓我們來自己試試看吧:
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,來動態(tài)的創(chuàng)建同名的方法。現(xiàn)在我們在類C里面使用這個聲明:generate_method :method1, :method2,當(dāng)然我們需要include模塊M。為什么ActiveRecord的model不需要include相關(guān)的模塊呢?當(dāng)然是因為Topic的父類ActiveRecord::Base已經(jīng)include了模塊Associations了。
類C通過include模塊M,調(diào)用了模塊M的一個included回調(diào)接口,讓類C去extend模塊G,換句話來說就是,通過include模塊M,來給類C動態(tài)添加一個類方法generate_method。
這個generate_method被定義在模塊G當(dāng)中,它接受一系列參數(shù),來動態(tài)創(chuàng)建相關(guān)的方法。于是我們就實現(xiàn)了這樣的DSL功能:
通過在類C里面聲明generate_method :method1, :method2,讓類C動態(tài)的添加了兩個實例方法method1,method2,是不是很有意思? 實際上rails的對象關(guān)聯(lián)聲明也是以同樣的方式實現(xiàn)的。