ruby宣告式語法的實現程式碼
在ActiveRecord可以用很方便的宣告方式來定義model之間的關聯關係,例如:
class Topic < ActiveRecord::Base
has_many :posts
belongs_to :user
end
has_many和belongs_to其實是Topic類的class method,標準寫法是:
class Topic < ActiveRecord::Base
Topic.has_many(:posts)
Topic.belongs_to(:user)
end
那麼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。
這個generate_method被定義在模組G當中,它接受一系列引數,來動態建立相關的方法。於是我們就實現了這樣的DSL功能:
透過在類C裡面宣告generate_method :method1, :method2,讓類C動態的添加了兩個例項方法method1,method2,是不是很有意思? 實際上rails的物件關聯宣告也是以同樣的方式實現的。
【ruby宣告式語法的實現程式碼】相關文章: