Переопределение оператора космического корабля Ruby's <=>

StackOverflow https://stackoverflow.com/questions/3058801

Вопрос

Я пытаюсь переопределить оператор 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
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top