3

短暂而甜蜜

在 Ruby 1.9 中运行此代码:

FOO = "global constant"

class Something
  FOO = "success!"

  def self.handle &block
    self.new.instance_eval &block
  end
end

class Other
  FOO = "wrong constant"

  def self.handle
    Something.handle{FOO}
  end
end

puts Something.handle{FOO}
puts Other.handle

我得到“成功!” 和“错误的常数”。我怎样才能让两个电话都打印“成功!”?这不是为了好玩而人为的练习——我不会为此浪费人们的时间。我有一个现实世界的问题,我已将其缩减为演示该问题的最简单的示例。继续阅读“为什么”。

更彻底的解释

调用 Something.handle{FOO} 可以正常工作。Ruby 使用 Something 类中给出的 FOO 定义。但是,如果我尝试从另一个类中以相同的方式调用它,它会为我提供该类的 FOO定义

我认为 Ruby 1.9 中更紧密的常量查找的想法是为了避免这样的问题。instance_eval 应该使用其接收者的范围(在本例中为 self.new),而不是调用块的范围。它适用于实例变量之类的东西,但不适用于常量。这不是优先级问题 - 删除“全局”和“错误”常量,ruby 仍然无法找到剩余的正确常量。

我遇到的现实问题是我有一个包含多个类的模块。我有一个接受块的方法,并在模块的上下文中运行该块。在该块内,我希望能够通过它们的短名称来引用这些类(就像我可以在模块本身的任何地方一样),而不是必须在模块名称前面加上。

这很痛苦:

ThirdPartyApis::MyAnswerSite.connection question.id, answer.id do |question_id, answer_id|
  question = ThirdPartyApis::MyAnswerSite::Question.find question_id
  answer = ThirdPartyApis::MyAnswerSite::Answer.find answer_id

  ThirdPartyApis::MyAnswerSite::Solution.new question, answer
end

这是令人愉快的:

ThirdPartyApis::MyAnswerSite.connection question.id, answer.id do |question_id, answer_id|
  question = Question.find question_id
  answer = Answer.find answer_id

  Solution.new question, answer
end

概括

这就是冗长的解释。请不要提供不能解决我的问题的解决方法。我很欣赏探索其他途径的愿望,但我的问题很简单:只能修改Something类,以打印“成功!”的方式 最后两次?这就是测试用例,任何适合这个的解决方案都是我接受的答案,它的作者是我本周的个人英雄。请!

4

1 回答 1

3

我使用 sourcecify gem ( gem install sourcify) 让它工作:

require 'sourcify'
FOO = "global constant"

class Something
  FOO = "success!"

  def self.handle &block
    (self.new.instance_eval(block.to_source)).call
  end

end

class Other
  FOO = "wrong constant"

  def self.handle
    Something.handle{FOO}
  end
end

puts Something.handle{FOO}
=> success!
puts Other.handle
=> success!

编辑:哎呀,你也可以看到我在哪里使用绑定,删除了该代码。

于 2011-02-03T23:56:10.760 回答