ruby - 更改 ruby​​ 方法上下文/使用 instance_exec 调用方法

标签 ruby methods module scope metaprogramming

首先,对于简短版本:

方法定义不就是一个 block 吗?为什么我不能做类似的事情:

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

目标是在单独类的实例上下文中运行某些模块方法?尽管调用了“instance_exec”,但该方法似乎并未在“obj”的上下文中执行。

我能弄清楚如何完成此操作的唯一方法是将“my_method”的所有代码包装在一个 proc 中,然后改为按以下方式调用:

obj.instance_eval(&other_obj.my_method)

但我想避免在 procs 中封装我所有的模块方法。


现在,对于长版本:

我正在尝试创建一个模块化的外部提供者系统,对于任何给定的类/方法(通常是 Controller 方法),我可以为给定的提供者(例如 facebook)调用相应的方法。

由于可能有多个提供者,提供者方法需要命名空间,但不是简单地包括一堆方法,例如“facebook_invitation_create”,我希望我的 InvitationsController 实例有一个包含创建方法 - 例如

class InvitationsController < ApplicationController
  def create
    ...
    # e.g. self.facebook.create
    self.send(params[:provider]).create
    ...
  end
end

此外,我希望提供者方法不仅可以像 Controller 本身的一部分一样运行——这意味着它们应该可以访问 Controller 实例变量、参数、 session 等——而且还可以 (大多数)编写得好像它们是 Controller 本身的一部分 - 这意味着由于模块化而没有任何复杂的附加代码。

我在下面创建了一个简化的示例,其中 MyClass 有一个 greet 方法,如果使用有效的提供者名称(在本例中为:facebook)调用该方法,则会改为调用该提供者的 greet 方法。反过来,提供者 greet 方法访问包含类的消息方法,就好像它是类本身的一部分一样。

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 吗?

(我也乐于接受有关如何更好地实现它的建议...)

最佳答案

我认为这比您预期的要简单(而且我意识到我的回答是在您提出问题 2 年后)

您可以使用模块中的实例方法并将它们绑定(bind)到任何对象。

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 并将其绑定(bind)到当前的 self.

这是委托(delegate)。 它是 casting 的基础gem 会像这样工作:

delegate(:greet, provider)

或者,如果您选择在转换中使用 method_missing,您的代码可能如下所示:

greet

但是你需要先设置你的委托(delegate):

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 中写的内容有关

关于ruby - 更改 ruby​​ 方法上下文/使用 instance_exec 调用方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15558472/

相关文章:

javascript - 浏览器中的 ES6 模块 : Uncaught SyntaxError: Unexpected token import

javascript - Angular js - 指定 Rails api 的主机名

ruby - Ruby 如何允许方法参数默认值从早期参数派生?

ios - subview 访问父 View 方法

java - 访问泛型类的静态方法

python - 如何检查请求帖子是否已成功发布?

ROS catkin 工作区中的 Python 模块和子模块

ruby-on-rails - ruby:如何有效地迭代哈希中的元素

javascript - 将 javascript 图表导出为 pdf

ruby - 如何在 Ruby 中实现查找类?