3

我有一个公开两个接口方法的类client_optionsuser_options并且在这个祖先级别上,它们等同于default_options. 我不希望其他开发人员default_options直接实现,因此它是私有的。

class Foo
  def client_options
    default_options
  end
  def user_options
    default_options
  end

  private
    def default_options
      { foo: :bar }
    end
end

为了节省一些代码行,我想给这些方法起别名:

class Foo
  alias_method :client_options, :default_options
  alias_method :user_options, :default_options

  private
    def default_options
      { foo: :bar }
    end
end

alias_method只有别名公共方法。

我在这个博客上找到了如何给私有方法起别名:

class Foo
  def default_options
    { foo: :bar}
  end

  private :default_options
  alias_method :client_options, :default_options
  public :client_options
end

但是,它有点不可读。

是否有更直接的解决方案来别名私有方法?

4

3 回答 3

10

别名,然后私有化:

alias_method :client_options, :default_options
alias_method :user_options, :default_options
private :default_options

或者,您是否非常了解这个“剪刀规则”:

%i(client_options user_options).each do |m|
  define_method m { default_options }
end

或创建自己的alias_method类似方法

  module AliasPrivateMethod
    def alias_private_method_to_interface(name, original_name)
      define_method(name) do |*args, &block|
        send(original_name, *args, &block)
      end
    end
  end

  class Foo
    extend AliasPrivateMethod
    alias_private_method_to_interface(:client_options, :default_options)
    private
      def default_options
        { foo: :bar }
      end
  end

  foo = Foo.new
  foo.public_methods(false) # => [:client_options]
  foo.client_options        # => { foo: :bar }
于 2015-08-13T08:50:04.933 回答
1

如何实现您想要隐藏在前置模块中的方法?

module ProtoFoo
  protected
    def default_options
      {foo: :bar}
    end
end

class Foo
  prepend ProtoFoo
  def client_options; default_options end
  def user_options; default_options end
end

即使用户在 中覆盖default_optionsFoo它也不会生效。

如果你坚持在暴露的东西之后写隐藏的东西,你可以这样做:

class Foo
  def client_options; default_options end
  def user_options; default_options end
end

module ProtoFoo
  def default_options
    {foo: :bar}
  end
end
Foo.prepend ProtoFoo
于 2015-08-13T09:15:36.013 回答
1

我发现一种很好的替代方法是将私有方法委托给self

require 'forwardable'
class Foo
  extend Forwardable
  def_delegator :self, :default_options, :client_options
  def_delegator :self, :default_options, :user_options

  private
    def default_options
      { foo: :bar }
    end
end

f = Foo.new
f.client_options 
# => { foo: :bar }
于 2015-08-13T08:47:43.633 回答