背景:

我有一个模块,它声明了许多实例方法

module UsefulThings
  def get_file; ...
  def delete_file; ...

  def format_text(x); ...
end

我想在类中调用其中一些方法。在ruby中你通常是这样做的:

class UsefulWorker
  include UsefulThings

  def do_work
    format_text("abc")
    ...
  end
end

问题

include UsefulThings引入了UsefulThings中的所有方法。在这种情况下,我只需要format_text和显式不需要get_file和delete_file。

我可以看到几个可能的解决方案:

Somehow invoke the method directly on the module without including it anywhere I don't know how/if this can be done. (Hence this question) Somehow include Usefulthings and only bring in some of it's methods I also don't know how/if this can be done Create a proxy class, include UsefulThings in that, then delegate format_text to that proxy instance This would work, but anonymous proxy classes are a hack. Yuck. Split up the module into 2 or more smaller modules This would also work, and is probably the best solution I can think of, but I'd prefer to avoid it as I'd end up with a proliferation of dozens and dozens of modules - managing this would be burdensome

为什么一个模块中有很多不相关的功能?它是来自rails应用程序的ApplicationHelper,我们的团队实际上已经决定将它作为任何不够特定而不属于其他任何地方的东西的垃圾场。大多数是独立的实用程序方法,在任何地方都可以使用。我可以把它分解成独立的助手,但有30个,每个都有一个方法……这似乎没有什么效果


当前回答

我认为丢弃单个调用(不改变现有模块或创建新模块)的最短方法如下:

Class.new.extend(UsefulThings).get_file

其他回答

今天我在学习Ruby的时候,发现这个老问题很有趣,所以我想用我的新知识来回答。

假设您拥有该模块

module MyModule
  def say
    'I say'
  end

  def cheer
    'I cheer'
  end
end 

然后在类中调用Animal,我可以从MyModule中采用cheer方法,如下所示

class Animal
  define_method(:happy, MyModule.method(:cheer))
end

这就是所谓的unbound方法,所以你可以把一个可调用的对象绑定到另一个地方。

从这里开始,您可以像往常一样使用该方法,例如

my_dog = Animal.new

my_dog.happy # => "I cheer"

希望这对你有所帮助,因为我今天也学到了一些新东西。

为了进一步了解,您可以使用irb并查看Method对象。

首先,我建议将模块分解为您需要的有用内容。但是你总是可以创建一个类来扩展你的调用:

module UsefulThings
  def a
    puts "aaay"
  end
  def b
    puts "beee"
  end
end

def test
  ob = Class.new.send(:include, UsefulThings).new
  ob.a
end

test

如果你想调用这些方法而不将module包含在另一个类中,那么你需要将它们定义为模块方法:

module UsefulThings
  def self.get_file; ...
  def self.delete_file; ...

  def self.format_text(x); ...
end

然后你就可以打电话给他们

UsefulThings.format_text("xxx")

or

UsefulThings::format_text("xxx")

但无论如何,我建议你把相关的方法放在一个模块或一个类中。如果你只想从模块中包含一个方法,那么这听起来就像是一种糟糕的代码味道,将不相关的方法放在一起不是好的Ruby风格。

我认为丢弃单个调用(不改变现有模块或创建新模块)的最短方法如下:

Class.new.extend(UsefulThings).get_file

根据我对这个问题的理解,您希望将模块的一些实例方法混合到一个类中。

让我们首先考虑Module#include是如何工作的。假设我们有一个UsefulThings模块,它包含两个实例方法:

module UsefulThings
  def add1
    self + 1
  end
  def add3
    self + 3
  end
end

UsefulThings.instance_methods
  #=> [:add1, :add3]

和Fixnum包括该模块:

class Fixnum
  def add2
    puts "cat"
  end
  def add3
    puts "dog"
  end
  include UsefulThings
end

我们看到:

Fixnum.instance_methods.select { |m| m.to_s.start_with? "add" }
  #=> [:add2, :add3, :add1] 
1.add1
2 
1.add2
cat
1.add3
dog

您是否期望UsefulThings#add3覆盖Fixnum#add3,以便1。Add3返回4?考虑一下:

Fixnum.ancestors
  #=> [Fixnum, UsefulThings, Integer, Numeric, Comparable,
  #    Object, Kernel, BasicObject] 

当类包含模块时,模块就成为类的超类。因此,由于继承的工作方式,将add3发送给Fixnum的实例将导致Fixnum#add3被调用,返回dog。

现在让我们添加一个方法:add2到UsefulThings:

module UsefulThings
  def add1
    self + 1
  end
  def add2
    self + 2
  end
  def add3
    self + 3
  end
end

我们现在希望Fixnum只包含add1和add3方法。这样做,我们期望得到与上面相同的结果。

假设,如上所述,我们执行:

class Fixnum
  def add2
    puts "cat"
  end
  def add3
    puts "dog"
  end
  include UsefulThings
end

结果是什么?不需要的方法:add2被添加到Fixnum,:add1被添加,由于上面解释的原因,:add3没有被添加。我们要做的就是undef:add2。我们可以用一个简单的helper方法来做到这一点:

module Helpers
  def self.include_some(mod, klass, *args)
    klass.send(:include, mod)
    (mod.instance_methods - args - klass.instance_methods).each do |m|
      klass.send(:undef_method, m)
    end
  end
end

我们这样调用:

class Fixnum
  def add2
    puts "cat"
  end
  def add3
    puts "dog"
  end
  Helpers.include_some(UsefulThings, self, :add1, :add3)
end

然后:

Fixnum.instance_methods.select { |m| m.to_s.start_with? "add" }
  #=> [:add2, :add3, :add1] 
1.add1
2 
1.add2
cat
1.add3
dog

这就是我们想要的结果。