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