加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

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

发布时间:2020-12-17 03:04:57 所属栏目:百科 来源:网络整理
导读:首先,对于短版本: 方法定义不仅仅是一个块吗?为什么我不能这样做: obj.instance_exec(other_obj.method(:my_method)) 目标是在单独的类的实例的上下文中运行一些模块方法?该方法被调用,但它似乎没有在’obj’的上下文中执行,尽管有’instance_exec’调用
首先,对于短版本:

方法定义不仅仅是一个块吗?为什么我不能这样做:

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

目标是在单独的类的实例的上下文中运行一些模块方法?该方法被调用,但它似乎没有在’obj’的上下文中执行,尽管有’instance_exec’调用.

我可以弄清楚如何实现这一点的唯一方法是将’my_method’的所有代码包装在一个proc中,然后以下面的方式调用:

obj.instance_eval(&other_obj.my_method)

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

现在,对于长版本:

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

由于可能有多个提供者,提供者方法需要命名空间,但不是简单地包含一堆方法,例如’facebook_invitation_create’,我希望我的InvitationsController实例拥有一个包含create方法的facebook成员 – 例如

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

此外,我希望提供者方法不仅像它们是控制器本身的一部分那样起作用 – 意味着它们应该能够访问诸如控制器实例变量,参数,会话等的东西 – 而且还要(大部分)写入好像它们是控制器本身的一部分 – 意味着没有任何复杂的附加代码作为模块化的结果.

我在下面创建了一个简化示例,其中MyClass有一个greet方法,如果使用有效的提供者名称(在这种情况下为facebook)调用,则会调用该提供者greet方法.反过来,provider 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年后)

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

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

如果您的提供者是一个模块,您可以使用instance_method创建一个UnboundMethod并将其绑定到当前的self.

这是代表团.
它是casting gem的基础,它可以像这样工作:

delegate(:greet,provider)

或者,如果您选择从强制转换中使用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写的内容有关

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读