首先,对于简短版本:
方法定义不只是一个块吗?为什么我不能这样做:
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 实例有一个包含创建方法的 facebook 成员 -例如
class InvitationsController < ApplicationController
def create
...
# e.g. self.facebook.create
self.send(params[:provider]).create
...
end
end
此外,我希望提供者方法不仅可以像控制器本身的一部分一样运行——这意味着它们应该可以访问控制器实例变量、参数、会话等内容——而且还可以(大部分)编写就好像它们是控制器本身的一部分一样——这意味着由于模块化而没有任何复杂的附加代码。
我在下面创建了一个简化的示例,其中 MyClass 有一个 greet 方法,如果使用有效的提供者名称(在本例中为:facebook)调用该方法,则会改为调用该提供者的 greet 方法。反过来,提供者 greet 方法访问包含类的 message 方法,就好像它是类本身的一部分一样。
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 ?
(我也愿意接受有关如何更好地实施这一点的建议......)