Переопределение оператора космического корабля Ruby's <=>
-
27-09-2019 - |
Вопрос
Я пытаюсь переопределить оператор Ruby's <=> (космический корабль), чтобы сортировать яблоки и апельсины, чтобы яблоки впервые отсортировали по весу, а апельсины вторые, отсортированные по сладости. Вот так:
module Fruity
attr_accessor :weight, :sweetness
def <=>(other)
# use Array#<=> to compare the attributes
[self.weight, self.sweetness] <=> [other.weight, other.sweetness]
end
include Comparable
end
class Apple
include Fruity
def initialize(w)
self.weight = w
end
end
class Orange
include Fruity
def initialize(s)
self.sweetness = s
end
end
fruits = [Apple.new(2),Orange.new(4),Apple.new(6),Orange.new(9),Apple.new(1),Orange.new(22)]
p fruits
#should work?
p fruits.sort
Но это не работает, может кто-нибудь сказать, что я делаю неправильно здесь, или лучший способ сделать это?
Решение
Ваша проблема заключается в том, что вы только инициализируете одну из свойств по любой стороне, другой все еще будет nil
. nil
не обрабатывается в Array#<=>
Метод, который заканчивается убийством своего рода.
Есть несколько способов справиться с проблемой, сначала будет что-то вроде этого
[self.weight.to_i, self.sweetness.to_i] <=> [other.weight.to_i, other.sweetness.to_i]
nil.to_i
дает тебе 0
, что позволит эту работу.
Другие советы
Вероятно поздно, тем не менее ...
Добавьте следующий обезьян
class Array
def to_i(default=Float::INFINITY)
self.map do |element|
element.nil? ? default : element.to_i
end
end
end
И изменить тело Fruity::<=> to
[self.weight, self.sweetness].to_i <=> [other.weight, other.sweetness].to_i