Как я могу вызвать Proc, который принимает блок в другом контексте? - PullRequest
13 голосов
/ 26 марта 2012

Возьмем этот пример Proc:

proc = Proc.new {|x,y,&block| block.call(x,y,self.instance_method)}

Он принимает два аргумента, x и y, а также блок.

Я хочу выполнить этот блок, используя разные значения для себя.Что-то вроде этого почти работает:

some_object.instance_exec("x arg", "y arg", &proc)

Однако это не позволяет вам проходить в блоке.Это также не работает

some_object.instance_exec("x arg", "y arg", another_proc, &proc)

и

some_object.instance_exec("x arg", "y arg", &another_proc, &proc)

Я не уверен, что еще может работать здесь.Возможно ли это, и если да, то как вы это делаете?

Редактировать: В основном, если вы можете передать этот файл rspec, изменив метод change_scope_of_proc, вы решили мою проблему.

require 'rspec'

class SomeClass
  def instance_method(x)
    "Hello #{x}"
  end
end

class AnotherClass
  def instance_method(x)
    "Goodbye #{x}"
  end

  def make_proc
    Proc.new do |x, &block|
      instance_method(block.call(x))
    end
  end
end

def change_scope_of_proc(new_self, proc)
  # TODO fix me!!!
  proc
end

describe "change_scope_of_proc" do
  it "should change the instance method that is called" do
    some_class = SomeClass.new
    another_class = AnotherClass.new
    proc = another_class.make_proc
    fixed_proc = change_scope_of_proc(some_class, proc)
    result = fixed_proc.call("Wor") do |x|
      "#{x}ld"
    end

    result.should == "Hello World"
  end
end

Ответы [ 2 ]

8 голосов
/ 29 марта 2012

Чтобы решить эту проблему, необходимо повторно привязать Proc к новому классу.

Вот ваше решение, использующее хороший код из Rails core_ext:

require 'rspec'

# Same as original post

class SomeClass
  def instance_method(x)
    "Hello #{x}"
  end
end

# Same as original post

class AnotherClass
  def instance_method(x)
    "Goodbye #{x}"
  end

  def make_proc
    Proc.new do |x, &block|
      instance_method(block.call(x))
    end
  end
end

### SOLUTION ###

# From activesupport lib/active_support/core_ext/kernel/singleton_class.rb

module Kernel
  # Returns the object's singleton class.
  def singleton_class
    class << self
      self
    end
  end unless respond_to?(:singleton_class) # exists in 1.9.2

  # class_eval on an object acts like singleton_class.class_eval.
  def class_eval(*args, &block)
    singleton_class.class_eval(*args, &block)
  end
end

# From activesupport lib/active_support/core_ext/proc.rb 

class Proc #:nodoc:
  def bind(object)
    block, time = self, Time.now
    object.class_eval do
      method_name = "__bind_#{time.to_i}_#{time.usec}"
      define_method(method_name, &block)
      method = instance_method(method_name)
      remove_method(method_name)
      method
    end.bind(object)
  end
end

# Here's the method you requested

def change_scope_of_proc(new_self, proc)
  return proc.bind(new_self)
end

# Same as original post

describe "change_scope_of_proc" do
  it "should change the instance method that is called" do
    some_class = SomeClass.new
    another_class = AnotherClass.new
    proc = another_class.make_proc
    fixed_proc = change_scope_of_proc(some_class, proc)
    result = fixed_proc.call("Wor") do |x|
      "#{x}ld"
    end
    result.should == "Hello World"
  end
end
0 голосов
/ 27 марта 2012

Я не думаю, что вы можете сделать это, и проблема не в том, чтобы пройти несколько блоков. Proc's и блоки являются замыканиями и фиксируют их привязки в точке создания. self является частью этой привязки, поэтому, даже если вы измените self с помощью instance_eval, когда вы call proc / block, он выполняет в привязке, а self закрывается:

$ irb
irb(main):001:0> class Foo; def mkproc; Proc.new { puts "#{self.class}:#{object_id}" }; end; end
=> nil
irb(main):002:0> p = Foo.new.mkproc
=> #<Proc:0x00000001b04338@(irb):1>
irb(main):003:0> p.call
Foo:14164520
=> nil
irb(main):004:0> 'bar'.instance_exec { puts "#{self.class}:#{object_id}"; p.call }
String:16299940
Foo:14164520

Ruby позволит вам захватить привязку замыкания с Kernel # привязка , но не предлагает способа установить привязку, связанную с Proc . Вы можете указать привязку для строковой версии Kernel # eval , но это по-прежнему не позволяет изменить привязку вызываемого вами процесса.

irb(main):005:0> class BindMe; def get_binding(p=nil); binding; end; end
=> nil
irb(main):006:0> b = BindMe.new.get_binding(p)
=> #<Binding:0x00000001f58e48>
irb(main):007:0> eval '"#{self.class}:#{object_id}"', b
=> "BindMe:14098300"
irb(main):008:0> eval '"#{self.class}:#{object_id}"', p.binding
=> "Foo:14164520"
irb(main):009:0> eval "p.call", b
Foo:14164520
...