respond_to 的 Ruby 细化问题?和范围

Posted

技术标签:

【中文标题】respond_to 的 Ruby 细化问题?和范围【英文标题】:Ruby refinement issues with respond_to? and scoping 【发布时间】:2015-11-05 05:15:50 【问题描述】:

我正在尝试将实例方法 foo 添加到 Ruby 的 Array 类 所以当它被调用时,数组的字符串元素会变成字符串“foo”。

这可以通过猴子修补 Ruby 的 StringArray 类轻松完成。

class String
  def foo
    replace "foo"
  end
end

class Array
  def foo
    self.each |x| x.foo if x.respond_to? :foo 
  end
end

a = ['a', 1, 'b']
a.foo
puts a.join(", ")   # you get 'foo, 1, foo' as expected

现在我正在尝试使用 Ruby 2 的 refinements feature 重写上述内容。 我使用的是 Ruby 版本 2.2.2。

以下工作(在文件中,例如 ruby​​ test.rb,但由于某种原因不在 irb 中)

module M
  refine String do
    def foo
      replace "foo"
    end
  end
end

using M
s = ''
s.foo
puts s      # you get 'foo'

但是,将 foo 添加到 Array 类时,我无法让它工作。

module M
  refine String do
    def foo
      replace "foo"
    end
  end
end

using M

module N
  refine Array do
    def foo
      self.each |x| x.foo if x.respond_to? :foo 
    end
  end
end

using N

a = ['a', 1, 'b']
a.foo
puts a.join(", ")   # you get 'a, 1, b', not 'foo, 1, foo' as expected

有两个问题:

    使用新方法优化类后,respond_to? 不起作用,即使您可以调用 对象上的方法。尝试添加puts 'yes' if s.respond_to? :foo 作为第二个代码 sn-p 的最后一行,您会看到 not 打印了“yes”。 在我的数组优化中,String#foo 超出了范围。如果你删除 if x.respond_to? :foo 从 Array#foo,你会得到错误undefined method 'foo' for "a":String (NoMethodError)。所以问题是:如何使 String#foo 细化在 Array#foo 细化中可见?

我如何克服这两个问题才能让它发挥作用?

(请不要提供不涉及细化的替代解决方案,因为这是一个理论练习,因此我可以学习如何使用细化)。

谢谢。

【问题讨论】:

【参考方案1】:

    respond_to? 方法不起作用,这已记录在案 here.

    问题是您只能在顶层激活细化 它们在范围上是词法的。

一种解决方案是:

module N
  refine String do
    def foo
      replace 'foobar'
    end
  end

  refine Array do
    def foo
      self.each do |x|
        x.foo rescue x
      end
    end
  end
end

using N

a = ['a', 1, 'b']
p a.foo

puts a.join(", ") # foo, 1, foo

【讨论】:

【参考方案2】:

再次以您的示例为例,一个简单的解决方案可能是覆盖 respond_to?细化块中的方法:

module M
  refine String do
    def foo
      replace "foo"
    end
    def respond_to?(name,all=false)
      list_methods = self.methods.concat [:foo]
      list_methods.include? name
    end
  end

  refine Array do
    def foo
      self.each |x| x.foo if x.respond_to? :foo 
    end
  end

end

using M

a = ['a', 1, 'b']
a.foo
puts a.join(", ")    # you get 'foo, 1, foo'

【讨论】:

以上是关于respond_to 的 Ruby 细化问题?和范围的主要内容,如果未能解决你的问题,请参考以下文章

Ruby的细化(refine关键字)

ruby 细化模式

Ruby 中的 method_missing 陷阱

`respond_to?` 与 `respond_to_missing?`

添加respond_to时的ActionController::UnknownFormat

Rails 5 API 控制器中未定义的实例方法“respond_to”