Применяются ли уточнения только к методам экземпляров?

Я пытаюсь понять функцию уточнения Руби, и я столкнулся со сценарием, который я не понимаю.

Возьмите этот пример кода:

class Traveller
  def what_are_you
    puts "I'm a Backpacker"
  end

  def self.preferred_accommodation
    puts "Hostels"
  end
end


module Refinements
  module Money
    def what_are_you
      puts "I'm a cashed-up hedonist!"
    end

    module ClassMethods
      def preferred_accommodation
        puts "Expensive Hotels"
      end
    end

    def self.included(base)
      base.extend ClassMethods
    end
  end

  refine Traveller do
    include Money
  end
end

Теперь, когда я делаю это в REPL:

Traveller.new.what_are_you         # => I'm a Backpacker
Traveller.preferred_accommodation  # => Hostels

using Refinements

Traveller.new.what_are_you         # => I'm a cashed-up hedonist!
Traveller.preferred_accommodation  # => Hostels (???)

Почему #what_are_you изысканный, но .preferred_accommodation не является?

2 ответа

Решение

Как объяснил @MasashiMiyazaki, вам нужно уточнить два класса: Traveller а также TravellerСинглтон-класс. Это на самом деле позволяет вам немного упростить ваш код:

module Money
  refine Traveller do
    def what_are_you
      puts "I'm a cashed-up hedonist!"
    end
  end

  refine Traveller.singleton_class do
    def preferred_accommodation
      puts "Expensive Hotels"
    end
  end
end

Traveller.new.what_are_you         #=> I'm a Backpacker
Traveller.preferred_accommodation  #=> Hostels

using Money
Traveller.new.what_are_you         #=> I'm a cashed-up hedonist!
Traveller.preferred_accommodation  #=> Expensive Hotels

Кроме того, поместив три вышеуказанных оператора в модуль, уточненные версии двух методов ограничиваются этим модулем:

module M
  using Money
  Traveller.new.what_are_you         #=> I'm a cashed-up hedonist!
  Traveller.preferred_accommodation  #=> Expensive Hotels
end

Traveller.new.what_are_you           #=> I'm a Backpacker
Traveller.preferred_accommodation    #=> Hostels

Вам нужно вызвать уточнение Traveler с областью действия singleton_class, чтобы перезаписать методы класса. Добавив следующий код в свой модуль уточнений вместо self.includedВы можете получить ожидаемый результат.

module Refinements
  refine Traveller.singleton_class do
    include Money::ClassMethods
  end
end

Эта статья ( http://timelessrepo.com/refinements-in-ruby) поможет вам лучше понять уточнения.

Другие вопросы по тегам