Поиск…


Синтаксис

  • 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


Modified text is an extract of the original Stack Overflow Documentation
Лицензировано согласно CC BY-SA 3.0
Не связан с Stack Overflow