ruby元編程之創(chuàng)建自己的動態(tài)方法
method_missing是Ruby元編程(metaprogramming)常用的手法?;舅枷胧峭ㄟ^實現(xiàn)調(diào)用不存在的方法,以便進行回調(diào)。典型的例子是:ActiveRecord的動態(tài)查找(dynamic finder)。例如:我們有email屬性那么就可以調(diào)用User.find_by_email('joe@example.com'),雖然, ActiveRecord::Base并沒有一個叫做find_by_email的方法。
respond_to? 并不如method_missing出名,常用在當需要確認一個回饋對象需要確認,以便不會因為沒有反饋對象,而導致后面的調(diào)用出現(xiàn)錯誤。
下面是一個應用這兩者的例子:
示例
我們有類Legislator class,現(xiàn)在,想要給它加一個find_by_first_name('John')的動態(tài)調(diào)用。實現(xiàn)find(:first_name => 'John')的功能。
class Legislator
#假設(shè)這是一個真實的實現(xiàn)
def find(conditions = {})
end
#在本身定義畢竟這是他的方法
def self.method_missing(method_sym, *arguments, &block)
# the first argument is a Symbol, so you need to_s it if you want to pattern match
if method_sym.to_s =~ /^find_by_(.*)$/
find($1.to_sym => arguments.first)
else
super
end
end
end
那么這個時候調(diào)用
Legislator.respond_to?(:find_by_first_name)
將會提示錯誤,那么繼續(xù)
class Legislator
# 省略
# It's important to know Object defines respond_to to take two parameters: the method to check, and whether to include private methods
# http://www.ruby-doc.org/core/classes/Object.html#M000333
def self.respond_to?(method_sym, include_private = false)
if method_sym.to_s =~ /^find_by_(.*)$/
true
else
super
end
end
end
正如代碼注釋所述respond_to?需要兩個參數(shù),如果,你沒有提供將會產(chǎn)生ArgumentError。
相關(guān)反射 DRY
如果我們注意到了這里有重復的代碼。我們可以參考ActiveRecord的實現(xiàn)封裝在ActiveRecord::DynamicFinderMatch,以便避免在method_missing和respond_to?中重復。
class LegislatorDynamicFinderMatch
attr_accessor :attribute
def initialize(method_sym)
if method_sym.to_s =~ /^find_by_(.*)$/
@attribute = $1.to_sym
end
end
def match?
@attribute != nil
end
end
class Legislator
def self.method_missing(method_sym, *arguments, &block)
match = LegislatorDynamicFinderMatch.new(method_sym)
if match.match?
find(match.attribute => arguments.first)
else
super
end
end
def self.respond_to?(method_sym, include_private = false)
if LegislatorDynamicFinderMatch.new(method_sym).match?
true
else
super
end
end
end
緩存 method_missing
重復多次的method_missing可以考慮緩存。
另外一個我們可以向ActiveRecord 學習的是,當定義method_missing的時候,發(fā)送 now-defined方法。如下:
class Legislator
def self.method_missing(method_sym, *arguments, &block)
match = LegislatorDynamicFinderMatch.new(method_sym)
if match.match?
define_dynamic_finder(method_sym, match.attribute)
send(method_sym, arguments.first)
else
super
end
end
protected
def self.define_dynamic_finder(finder, attribute)
class_eval <<-RUBY
def self.#{finder}(#{attribute}) # def self.find_by_first_name(first_name)
find(:#{attribute} => #{attribute}) # find(:first_name => first_name)
end # end
RUBY
end
end
測試
測試部分如下:
describe LegislatorDynamicFinderMatch do
describe 'find_by_first_name' do
before do
@match = LegislatorDynamicFinderMatch.new(:find_by_first_name)
end
it 'should have attribute :first_name' do
@match.attribute.should == :first_name
end
it 'should be a match' do
@match.should be_a_match
end
end
describe 'zomg' do
before do
@match = LegislatorDynamicFinderMatch(:zomg)
end
it 'should have nil attribute' do
@match.attribute.should be_nil
end
it 'should not be a match' do
@match.should_not be_a_match
end
end
end
下面是 RSpec 例子:
describe Legislator, 'dynamic find_by_first_name' do
it 'should call find(:first_name => first_name)' do
Legislator.should_receive(:find).with(:first_name => 'John')
Legislator.find_by_first_name('John')
end
end
相關(guān)文章
Ruby中require、load、include、extend的區(qū)別介紹
這篇文章主要介紹了Ruby中require、load、include、extend的區(qū)別介紹,require、load用于文件,如.rb等等結(jié)尾的文件,include、load則用于包含一個文件中的模塊,需要的朋友可以參考下2015-05-05Ruby連接使用windows下sql server數(shù)據(jù)庫代碼實例
這篇文章主要介紹了Ruby連接使用windows下sql server數(shù)據(jù)庫代碼實例,本文直接給出實現(xiàn)代碼,而且給出了兩種實現(xiàn)和access數(shù)據(jù)庫的實現(xiàn)代碼,需要的朋友可以參考下2015-05-05