欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

ruby元编程之创建自己的动态方法

程序员文章站 2022-04-28 22:25:45
method_missing是ruby元编程(metaprogramming)常用的手法。基本思想是通过实现调用不存在的方法,以便进行回调。典型的例子是:activerec...

method_missing是ruby元编程(metaprogramming)常用的手法。基本思想是通过实现调用不存在的方法,以便进行回调。典型的例子是:activerecord的动态查找(dynamic finder)。例如:我们有email属性那么就可以调用user.find_by_email('joe@example.com'),虽然, activerecord::base并没有一个叫做find_by_email的方法。

respond_to? 并不如method_missing出名,常用在当需要确认一个回馈对象需要确认,以便不会因为没有反馈对象,而导致后面的调用出现错误。

下面是一个应用这两者的例子:

示例

我们有类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