首先,对于短版本:
方法定义不仅仅是一个块吗?为什么我不能这样做:
obj.instance_exec(&other_obj.method(:my_method))
目标是在单独的类的实例的上下文中运行一些模块方法?该方法被调用,但它似乎没有在'obj'的上下文中执行,尽管有'instance_exec'调用。
我能弄清楚如何实现这一目标的唯一方法是在proc中包含'my_method'的所有代码,然后以下面的方式调用:
obj.instance_eval(&other_obj.my_method)
但我想避免在procs中封装我的所有模块方法。
现在,对于长版本:
我正在尝试创建一个模块化的外部提供程序系统,对于任何给定的类/方法(通常是控制器方法),我可以为给定的提供程序(例如facebook)调用相应的方法。
由于可能有多个提供者,提供者方法需要命名空间,但不是简单地包含一堆方法,例如'facebook_invitation_create',我希望我的InvitationsController实例拥有一个包含创建方法 - 例如
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?
(我愿意接受有关如何更好地实施此建议的建议......)
答案 0 :(得分:6)
我认为这比你想象的要简单(我意识到我的回答是你提出的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
如果您的provider
是一个模块,则可以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
中所写的内容有关答案 1 :(得分:0)
也许我没有跟进,但它似乎就像你正在努力使它变得更难。
为什么不在你的类中实现一个“dispatch”模式,你有一个提供者名称和提供者方法的哈希值{:facebook =&gt;“facebook_greet”}然后只是通过“发送”来电来自正确的处理程序“对象#发送”(http://ruby-doc.org/core-1.9.3/Object.html#method-i-send)?对于调度方法,发送非常快,因此与eval不同,您应该获得出色的性能。
这里有一些代码来演示我解决它的方式(假设我跟随你想要完成的事情):
module TwitterProvider
def providerInit(providers)
@providers[:twitter]="twitter_greet"
super(providers) if defined?(super)
end
def twitter_greet
"Hello Twitter User"
end
end
module FacebookProvider
def providerInit(providers)
providers[:facebook]="facebook_greet"
super(providers) if defined?(super)
end
def facebook_greet
"Hello Facebook User"
end
end
class MyClass
include FacebookProvider
include TwitterProvider
attr_accessor :message
def providerInit(providers)
super(providers) if defined?(super)
end
def initialize(message="hello")
@providers = {}
self.message = message
providerInit(@providers)
end
def greet(provider=nil)
if provider.nil? or !self.respond_to?(@providers[provider])
self.message
else
self.send(@providers[provider])
end
end
end
my_class = MyClass.new
puts my_class.greet
puts my_class.greet(:twitter)
puts my_class.greet(:facebook)
# Output:
# hello
# Hello Twitter User
# Hello Facebook User