最初的所謂Ruby元編程是一種可以在運行時動態地操縱語言結構的技術,您甚至可以輸入一個新的Ruby代碼并在運行時執行它,而無需重新啟動它,需要的朋友可以進入下文參考一下。
示例
我們有類Legislator class,現在,想要給它加一個find_by_first_name('John')的動態調用。實現find(:first_name => 'John')的功能。
復制代碼 代碼如下:
class Legislator
? #假設這是一個真實的實現
? 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
?
那么這個時候調用
?
復制代碼 代碼如下:
Legislator.respond_to?(:find_by_first_name)?
?
將會提示錯誤,那么繼續
復制代碼 代碼如下:
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?需要兩個參數,如果,你沒有提供將會產生ArgumentError。
相關反射 DRY
如果我們注意到了這里有重復的代碼。我們可以參考ActiveRecord的實現封裝在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的時候,發送 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
大家看完上文ruby元編程之創建自己的動態方法之后,知道要怎么操作了吧!總之,為了讓我們在工作中能順利,要多關注武林技術頻道分享的內容哦!
?