Ruby Language
Динамическая оценка
Поиск…
Синтаксис
- eval "source"
- eval "source", привязка
- eval "source", proc
- binding.eval "source" # равно
eval "source", binding
параметры
параметр | подробности |
---|---|
"source" | Любой исходный код Ruby |
binding | Экземпляр класса Binding |
proc | Экземпляр класса Proc |
Оценка экземпляра
Метод instance_eval
доступен для всех объектов. Он оценивает код в контексте получателя:
object = Object.new
object.instance_eval do
@variable = :value
end
object.instance_variable_get :@variable # => :value
instance_eval
устанавливает self
для object
в течение всего блока кода:
object.instance_eval { self == object } # => true
Приемник также передается блоку в качестве единственного аргумента:
object.instance_eval { |argument| argument == object } # => true
Метод instance_exec
отличается в этом отношении: вместо этого он передает свои аргументы блоку.
object.instance_exec :@variable do |name|
instance_variable_get name # => :value
end
Оценка строки
Любая String
может быть оценена во время выполнения.
class Example
def self.foo
:foo
end
end
eval "Example.foo" #=> :foo
Оценка внутри привязки
Ruby отслеживает локальные переменные и переменную self
через объект, называемый привязкой. Мы можем получить привязку области с вызовом Kernel#binding
и оценить строку внутри привязки через Binding#eval
.
b = proc do
local_variable = :local
binding
end.call
b.eval "local_variable" #=> :local
def fake_class_eval klass, source = nil, &block
class_binding = klass.send :eval, "binding"
if block
class_binding.local_variable_set :_fake_class_eval_block, block
class_binding.eval "_fake_class_eval_block.call"
else
class_binding.eval source
end
end
class Example
end
fake_class_eval Example, <<-BLOCK
def self.foo
:foo
end
BLOCK
fake_class_eval Example do
def bar
:bar
end
end
Example.foo #=> :foo
Example.new.bar #=> :bar
Динамическое создание методов из строк
Ruby предлагает define_method
как частный метод для модулей и классов для определения новых методов экземпляра. Тем не менее, «тело» метода должно быть Proc
или другим существующим методом.
Один из способов создания метода из необработанных строковых данных - использовать eval
для создания Proc из кода:
xml = <<ENDXML
<methods>
<method name="go">puts "I'm going!"</method>
<method name="stop">7*6</method>
</methods>
ENDXML
class Foo
def self.add_method(name,code)
body = eval( "Proc.new{ #{code} }" )
define_method(name,body)
end
end
require 'nokogiri' # gem install nokogiri
doc = Nokogiri.XML(xml)
doc.xpath('//method').each do |meth|
Foo.add_method( meth['name'], meth.text )
end
f = Foo.new
p Foo.instance_methods(false) #=> [:go, :stop]
p f.public_methods(false) #=> [:go, :stop]
f.go #=> "I'm going!"
p f.stop #=> 42