Уточняются ли только методы экземпляра?
Я пытаюсь понять функцию Ruby уточнения, и я столкнулся с сценарием, который я не понимаю.
Возьмите этот пример кода:
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
нет?
Ответы
Ответ 1
Как пояснил @MasashiMiyazaki, вам нужно уточнить два класса: класс Traveller
и Traveller
singleton. Это на самом деле позволяет вам немного упростить код:
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
Ответ 2
Чтобы перезаписать методы класса, вам необходимо вызвать уточнение Traveler с областью singleton_class. Добавив следующий код в модуль Refinements вместо self.included
, вы можете получить ожидаемый результат.
module Refinements
refine Traveller.singleton_class do
include Money::ClassMethods
end
end
Эта статья (http://timelessrepo.com/refinements-in-ruby) поможет вам лучше понять Refinements.