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_eval
或 instance_exec
使用本地变量而不是实例变量?
有没有我漏掉的一些概念?
为了评估局部变量,您需要传入字符串 `"local_var",它将 return 局部变量的值。如果你传入一个块,那么一个参数就不能传入,根据我的解释 the documentation.
块形式的实例 eval 的行为是作为闭包访问该调用所在对象的实例变量和私有方法。
带参数的实例 eval 的行为允许您计算该调用范围内的字符串。
经过我朋友的一些搜索和建议,我想我找到了问题所在。
在 ruby 中有两个 Context
当您的代码 运行 self
和 binding
时,当您使用 local vars
或 method
时没有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_eval
的 WHY
,因为它的文档说 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
。
我正在尝试一些 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_eval
或 instance_exec
使用本地变量而不是实例变量?
有没有我漏掉的一些概念?
为了评估局部变量,您需要传入字符串 `"local_var",它将 return 局部变量的值。如果你传入一个块,那么一个参数就不能传入,根据我的解释 the documentation.
块形式的实例 eval 的行为是作为闭包访问该调用所在对象的实例变量和私有方法。
带参数的实例 eval 的行为允许您计算该调用范围内的字符串。
经过我朋友的一些搜索和建议,我想我找到了问题所在。
在 ruby 中有两个 Context
当您的代码 运行 self
和 binding
时,当您使用 local vars
或 method
时没有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_eval
的 WHY
,因为它的文档说 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
。