In "Is it possible to sort a list of objects depending on if the individual object's response to a method?", I discovered that the flying saucer doesn't work on booleans.
Consider:
Ruby 1.8.7:
[true, false].sort # => undefined method `<=>' for true:TrueClass (NoMethodError)
true <=> false # => undefined method `<=>' for true:TrueClass (NoMethodError)
Ruby 1.9.3:
[true, false].sort # => comparison of TrueClass with false failed (ArgumentError)
true <=> false # => nil
true <=> true # => 0
false <=> true # => nil
It may have something to do with true and false not having a canonical sort order, because which comes first? But, that sounds pretty weak to me.
Is this a bug in sort
?
The Ruby language designer(s) probably felt that to invent an ordering for booleans would be a surprise to developers so they intentionally left out the comparison operators.
The so-called flying saucer requires all comparison operators (<
, >
, ==
) to work (not technically, although certainly theoretically). true
and false
are not less-than or greater-than each other. The same will hold true for nil
. For a practical workaround, you can 'cast' to integers (0 for false, 1 for true). Something like:
[true, false, true].sort_by{|e| e ? 1 : 0}
Booleans have no natural ordering. Unlike C, false is not less than true, they're just equivalent and equally valid states. However it is possible to configure the sort any way you like using a block, for example:
ary = [true, false, false, true]
ary.sort {|a,b| a == b ? 0 : a ? 1 : -1 }
# => [false, false, true, true]
Reversing the order is also trivial:
ary.sort {|a,b| a == b ? 0 : a ? -1 : 1 }
# => [true, true, false, false]
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With