Ruby 改进 instance_eval
Ruby refinements with instance_eval
我想对 DSL 进行一些改进。我能够使用此示例进行改进:
module ArrayExtras
refine Array do
def speak
puts 'array!'
end
end
end
module MyUniverse
using ArrayExtras
class Thing
def initialize
[1].speak
end
end
end
MyUniverse::Thing.new
这打印出来 "array!" 就好了。但是我一引入instance_eval
,方法就找不到了:
module MyUniverse
using ArrayExtras
class DSL
def initialize(&block)
instance_eval(&block)
end
end
end
MyUniverse::DSL.new do
[1].speak
end
我得到 undefined method
speak' for [1]:Array (NoMethodError)`
有没有办法在 instance_eval 中进行优化?
如果要细化ruby核心BaseObject
,需要修改如下。
module ArrayExtras
refine ::Array do
def speak
puts 'array!'
end
end
end
它将在顶层找到 class。
改进是词法范围内的。您在错误的词汇上下文中激活了优化。您需要在调用精化方法的地方激活它:
module ArrayExtras
refine Array do
def speak
puts 'array!'
end
end
end
module MyUniverse
class DSL
def initialize(&block)
instance_eval(&block)
end
end
end
using ArrayExtras
MyUniverse::DSL.new do
[1].speak
end
# array!
在某些情况下,您可以通过 using ArrayExtras
绑定来实现它。
module MyUniverse
using ArrayExtras
class DSL
def initialize(&block)
block.binding.eval("using ArrayExtras")
instance_eval(&block)
end
end
end
MyUniverse::DSL.new do
[1].speak
end
然而,这仅在您不在实例中使用您的 class 时有效,它仅在 binding
是模块或 class 上下文时有效,否则 eval
将失败,因为 main.using is permitted only at toplevel
.
我想对 DSL 进行一些改进。我能够使用此示例进行改进:
module ArrayExtras
refine Array do
def speak
puts 'array!'
end
end
end
module MyUniverse
using ArrayExtras
class Thing
def initialize
[1].speak
end
end
end
MyUniverse::Thing.new
这打印出来 "array!" 就好了。但是我一引入instance_eval
,方法就找不到了:
module MyUniverse
using ArrayExtras
class DSL
def initialize(&block)
instance_eval(&block)
end
end
end
MyUniverse::DSL.new do
[1].speak
end
我得到 undefined method
speak' for [1]:Array (NoMethodError)`
有没有办法在 instance_eval 中进行优化?
如果要细化ruby核心BaseObject
,需要修改如下。
module ArrayExtras
refine ::Array do
def speak
puts 'array!'
end
end
end
它将在顶层找到 class。
改进是词法范围内的。您在错误的词汇上下文中激活了优化。您需要在调用精化方法的地方激活它:
module ArrayExtras
refine Array do
def speak
puts 'array!'
end
end
end
module MyUniverse
class DSL
def initialize(&block)
instance_eval(&block)
end
end
end
using ArrayExtras
MyUniverse::DSL.new do
[1].speak
end
# array!
在某些情况下,您可以通过 using ArrayExtras
绑定来实现它。
module MyUniverse
using ArrayExtras
class DSL
def initialize(&block)
block.binding.eval("using ArrayExtras")
instance_eval(&block)
end
end
end
MyUniverse::DSL.new do
[1].speak
end
然而,这仅在您不在实例中使用您的 class 时有效,它仅在 binding
是模块或 class 上下文时有效,否则 eval
将失败,因为 main.using is permitted only at toplevel
.