首先,对于简短的版本:
方法定义不就是块吗?为什么我不能这样做:

obj.instance_exec(&other_obj.method(:my_method))

以在单独类的实例上下文中运行某个模块方法为目标?方法被调用,但它似乎没有在“obj”的上下文中执行,尽管调用了“instance_exec”。
我唯一能想到的方法是将“my_method”的所有代码包装在一个proc中,然后按以下方式调用:
obj.instance_eval(&other_obj.my_method)

但我希望避免在过程中封装所有模块方法。
现在,对于长版本:
我试图创建一个模块化的外部提供程序系统,其中对于任何给定的类/方法(通常是控制器方法),我可以为给定的提供程序(例如facebook)调用相应的方法。
由于可能有多个提供程序,提供程序方法需要有名称空间,但我希望invitationscontroller实例有一个包含create方法的facebook成员,而不是简单地包含一堆方法,例如“facebook邀请创建”。
class InvitationsController < ApplicationController
  def create
    ...
    # e.g. self.facebook.create
    self.send(params[:provider]).create
    ...
  end
end

此外,我希望provider方法不仅像控制器本身的一部分那样工作-这意味着它们应该可以访问控制器实例变量、参数、会话,等等,但也要(大多数情况下)像是控制器本身的一部分一样编写,这意味着模块化后没有任何复杂的附加代码。
我在下面创建了一个简单的示例,其中myClass有一个greet方法,如果使用有效的提供者名称(:facebook在本例中)调用它,它将改为调用该提供者greet方法。反过来,provider greet方法访问include类的消息方法,就好像它是类本身的一部分一样。
module Providers
  def facebook
    @facebook ||= FacebookProvider
  end

  module FacebookProvider
    class << self
      def greet
        proc {
          "#{message} from facebook!"
        }
      end
    end
  end
end

class MyClass
  include Providers
  attr_accessor :message

  def initialize(message="hello")
    self.message = message
  end

  def greet(provider=nil)
    (provider.nil? or !self.respond_to?(provider)) ? message : instance_exec(&self.send(provider).greet)
  end
end

这实际上完成了我之前所说的几乎所有事情,但是我对我的提供者函数需要封装在procs中这一事实感到很困惑。我想也许我可以直接调用方法上的instance_exec(在移除proc封装之后):
instance_exec(&self.send(provider).method(:greet))

…但当我得到错误消息时,似乎忽略了instance_exec:
NameError: undefined local variable or method `message' for Providers::FacebookProvider:Module

有没有办法在定义的方法上调用instance_exec?
(我对如何更好地实施这一点持开放态度……)

最佳答案

我认为这比你想象的要简单(我知道我的答案是在你问了两年之后)
可以使用模块中的实例方法并将它们绑定到任何对象。

module Providers
  def facebook
    @facebook ||= FacebookProvider
  end

  module FacebookProvider
    def greet
      "#{message} from facebook!"
    end
  end
end

class MyClass
  include Providers
  attr_accessor :message

  def initialize(message="hello")
    self.message = message
  end

  def greet(provider=nil)
    if provider
      provider.instance_method(:greet).bind(self).call
    else
      message
    end
  end
end

如果provider是一个模块,则可以使用instance_method创建UnboundMethod并将其绑定到当前self
这是代表团。
它是castinggem的基础,其工作原理如下:
delegate(:greet, provider)

或者,如果您选择使用casting中的method_missing,则您的代码可能如下所示:
greet

但你需要先设定你的代表:
class MyClass
  include Providers
  include Casting::Client
  delegate_missing_methods
  attr_accessor :message

  def initialize(message="hello", provider=facebook)
    cast_as(provider)
    self.message = message
  end
end

MyClass.new.greet # => "hello from facebook!"

我写的是what delegation is而不是on my blog这与理解dci和我在Clean Ruby中写的内容有关。

09-10 08:24
查看更多