ruby 访问 instance_eval 中的实例变量

ruby access instance variable in instance_eval

我正在尝试一些 ruby 元编程,但对 instance_eval() 感到有些困惑。

请参阅以下示例

@instance_var = 'instance_var'
local_var = 'local_var'
obj = Object.new
obj.instance_eval { p @instance_var; p local_var }
obj.instance_eval { @instance_var  = 'instance_var_in_obj'; local_var = 'local_var_in_obj' }
p @instance_var; p local_var

我希望@instance_var 和 local_var 都可以 pass/modify 在块中,但我得到了

nil
"local_var"
"instance_var"
"local_var_in_obj"

因此我们可以在 instance_val 中共享(pass/modify)本地变量,但实例变量属于 self 无法共享。

大约 instance_exec:

obj.instance_exec(@instance_var) {|instance_var| p instance_var; instance_var = @instance_var }
=> "instance_var"
@instance_var
=> "instance_var"

现在我可以传递我的外部实例变量,但仍然无法修改它。

@instance_arr = []
obj.instance_exec(@instance_arr) {|instance_arr| instance_arr << 'in_block' }
@instance_arr
=> ["in_block"]
obj.instance_exec(@instance_arr) {|instance_arr| instance_arr = [] }
@instance_arr
=> ["in_block"]

使用数组的实例变量我可以修改我的实例变量但只能在当前数组对象内

总结一下 instance_evalinstance_exec 使用本地变量而不是实例变量?

有没有我漏掉的一些概念?

为了评估局部变量,您需要传入字符串 `"local_var",它将 return 局部变量的值。如果你传入一个块,那么一个参数就不能传入,根据我的解释 the documentation.

块形式的实例 eval 的行为是作为闭包访问该调用所在对象的实例变量和私有方法。

带参数的实例 eval 的行为允许您计算该调用范围内的字符串。

经过我朋友的一些搜索和建议,我想我找到了问题所在。 在 ruby 中有两个 Context 当您的代码 运行 selfbinding 时,当您使用 local varsmethod 时没有set self.xxx 首先会检查它是否在你的 binding 对象中作为 local var 如果不是 Ruby 会认为它是一个方法然后搜索你的 self 对象找到它的定义并调用它。 想想看:

class A
  def test
    4
  end
  def use_variable
    test = 5
    test
  end
  def use_method
    test = 5
    self.test
  end
end
a = A.new
a.use_variable # returns 5
a.use_method   # returns 4

这解释了 instance_evalWHY,因为它的文档说 instance_eval 只是在给定的块中更改 self 而不是触及 binding 所以方法将是搜索新 self,本地值仍在同一 binding 对象中。

关于 instance_exec 我对此不是很确定,但看起来像实例变量(带有 at 前缀变量)它将在 self 上搜索,直接跳过 binding,所以在 instance_exec 中你的 @instance_arr 属于旧的 self 而在 instance_exec 块中你将它作为新的 local var 在新的 binding 块中得到(块有自己的范围)但它的值实际上是 @instance_arr 的引用,所以在新的 local var 上调用方法,比如 push 它会改变它们,因为它们共享相同的 Array instance,但是当您将新的 Array instance 分配给新的 local var 时,它们不再引用相同的 Array instance,那是第二个 WHY