Ruby setter 不适用于 eval



我在initialize方法中使用eval()

class ActiveRecord::FakedModel
  def initialize(attributes={})
    attributes = {} if not attributes
    attributes.each do |attr_name, value|
      eval("@#{attr_name}=#{value.inspect}")
    end
    @attributes = attributes
  end
...
end

并有一个用于清理空格的设置器:

class ContactForm < ActiveRecord::FakedModel
  attr_accessor :plz
  def plz=(plz)
    @plz = plz.try(:delete,' ')
  end
...
end

但是当我在哈希中给出"请"时,这个二传手不起作用:

c=ContactForm.new(:plz=>' 1  3 3 3 ')
=> #<ContactForm:0x1024e3a10 @attributes={:plz=>" 1  3 3 3 "}, @plz=" 1  3 3 3 ">

eval中使用二传手有什么问题吗?

您的 eval 语句没有调用 setter 方法,而是直接设置实例变量。如果希望构造函数使用资源库,请使用 send:

attributes.each do |attr_name, value|
  send "#{attr_name}=", value
end

使用 Object#instance_variable_set 方法设置实例变量。

attributes.each do |attr_name, value|
  self.instance_variable_set("@#{attr_name}", value.inspect)
end

若要动态执行方法,请使用 Object#send

class ActiveRecord::FakedModel
  def initialize(attributes={})
    attributes = {} if not attributes
    attributes.each do |attr_name, value|
      send("#{attr_name}=", value)
    end
    @attributes = attributes
  end
end

您还可以获得无需调用 inspect 并强制转换为变量字符串的优势。

您也可以使用 Object#instance_variable_set ,但在这种情况下,您将绕过 setter 方法,并且如果 setter 中有一些自定义逻辑(例如强制转换),则代码将无法按预期工作。

class ActiveRecord::FakedModel
  def initialize(attributes={})
    attributes = {} if not attributes
    attributes.each do |attr_name, value|
      instance_variable_set("@#{attr_name}", value)
    end
    @attributes = attributes
  end
end

最新更新