summaryrefslogtreecommitdiffhomepage
path: root/mrbgems/mruby-rational
diff options
context:
space:
mode:
authorKOBAYASHI Shuji <[email protected]>2019-05-16 22:09:56 +0900
committerKOBAYASHI Shuji <[email protected]>2019-05-16 22:09:56 +0900
commitf5fb1307b017fb972c12b4ec4b1866d789b0ca09 (patch)
tree944be2d50b5d40da236cd81d3793bad4884c4aca /mrbgems/mruby-rational
parent9b604abc8300d378bb32fb0e88d6f0831ed7e0d9 (diff)
downloadmruby-f5fb1307b017fb972c12b4ec4b1866d789b0ca09.tar.gz
mruby-f5fb1307b017fb972c12b4ec4b1866d789b0ca09.zip
Fix `Rational#==`
Diffstat (limited to 'mrbgems/mruby-rational')
-rw-r--r--mrbgems/mruby-rational/mrbgem.rake1
-rw-r--r--mrbgems/mruby-rational/mrblib/rational.rb29
-rw-r--r--mrbgems/mruby-rational/test/rational.rb49
3 files changed, 72 insertions, 7 deletions
diff --git a/mrbgems/mruby-rational/mrbgem.rake b/mrbgems/mruby-rational/mrbgem.rake
index 93f5b601c..496082709 100644
--- a/mrbgems/mruby-rational/mrbgem.rake
+++ b/mrbgems/mruby-rational/mrbgem.rake
@@ -3,7 +3,6 @@ MRuby::Gem::Specification.new('mruby-rational') do |spec|
spec.author = 'mruby developers'
spec.summary = 'Rational class'
- spec.add_dependency 'mruby-metaprog', core: 'mruby-metaprog'
spec.add_dependency 'mruby-object-ext', core: 'mruby-object-ext'
spec.add_dependency 'mruby-numeric-ext', core: 'mruby-numeric-ext'
end
diff --git a/mrbgems/mruby-rational/mrblib/rational.rb b/mrbgems/mruby-rational/mrblib/rational.rb
index 7d5b87362..dbc855af0 100644
--- a/mrbgems/mruby-rational/mrblib/rational.rb
+++ b/mrbgems/mruby-rational/mrblib/rational.rb
@@ -1,4 +1,7 @@
class Rational < Numeric
+ # Override #<, #<=, #>, #>= in Numeric
+ prepend Comparable
+
def initialize(numerator = 0, denominator = 1)
@numerator = numerator
@denominator = denominator
@@ -66,6 +69,24 @@ class Rational < Numeric
end
end
+ def <=>(rhs)
+ case rhs
+ when Fixnum
+ return @numerator <=> rhs if @denominator == 1
+ rhs = Rational(rhs)
+ when Float
+ return to_f <=> rhs
+ end
+ case rhs
+ when Rational
+ (@numerator * rhs.denominator - @denominator * rhs.numerator) <=> 0
+ when Numeric
+ return rhs <=> self
+ else
+ nil
+ end
+ end
+
def negative?
numerator.negative?
end
@@ -86,17 +107,17 @@ def Rational(numerator = 0, denominator = 1)
end
[Fixnum, Float].each do |cls|
- [:+, :-, :*, :/, :==].each do |op|
+ [:+, :-, :*, :/, :<=>, :==, :<, :<=, :>, :>=].each do |op|
cls.instance_exec do
original_operator_name = "__original_operator_#{op}_rational"
alias_method original_operator_name, op
define_method op do |rhs|
if rhs.is_a? Rational
- Rational(self).send(op, rhs)
+ Rational(self).__send__(op, rhs)
else
- send(original_operator_name, rhs)
+ __send__(original_operator_name, rhs)
end
end
end
end
-end \ No newline at end of file
+end
diff --git a/mrbgems/mruby-rational/test/rational.rb b/mrbgems/mruby-rational/test/rational.rb
index 85cebc316..4d3d36ccc 100644
--- a/mrbgems/mruby-rational/test/rational.rb
+++ b/mrbgems/mruby-rational/test/rational.rb
@@ -1,8 +1,18 @@
def assert_rational(real, exp)
- assert_float real.numerator, exp.numerator
+ assert_float real.numerator, exp.numerator
assert_float real.denominator, exp.denominator
end
+def assert_equal_rational(exp, r1, r2)
+ if exp
+ assert_operator(r1, :==, r2)
+ assert_not_operator(r1, :!=, r2)
+ else
+ assert_not_operator(r1, :==, r2)
+ assert_operator(r1, :!=, r2)
+ end
+end
+
assert 'Rational' do
r = 5r
assert_equal Rational, r.class
@@ -54,4 +64,39 @@ assert 'Rational#/' do
assert_rational Rational(-2, 9) / Rational(-9, 2), Rational(4, 81)
assert_rational Rational(9, 8) / 4, Rational(9, 32)
assert_float Rational(20, 9) / 9.8, 0.22675736961451246
-end \ No newline at end of file
+end
+
+assert 'Rational#==, Rational#!=' do
+ assert_equal_rational(true, Rational(1,1), Rational(1))
+ assert_equal_rational(true, Rational(-1,1), -1r)
+ assert_equal_rational(true, Rational(13,4), 3.25)
+ assert_equal_rational(true, Rational(13,3.25), Rational(4,1))
+ assert_equal_rational(true, Rational(-3,-4), Rational(3,4))
+ assert_equal_rational(true, Rational(-4,5), Rational(4,-5))
+ assert_equal_rational(true, Rational(4,2), 2)
+ assert_equal_rational(true, Rational(-4,2), -2)
+ assert_equal_rational(true, Rational(4,-2), -2)
+ assert_equal_rational(true, Rational(4,2), 2.0)
+ assert_equal_rational(true, Rational(-4,2), -2.0)
+ assert_equal_rational(true, Rational(4,-2), -2.0)
+ assert_equal_rational(true, Rational(8,6), Rational(4,3))
+ assert_equal_rational(false, Rational(13,4), 3)
+ assert_equal_rational(false, Rational(13,4), 3.3)
+ assert_equal_rational(false, Rational(2,1), 1r)
+ assert_equal_rational(false, Rational(1), nil)
+ assert_equal_rational(false, Rational(1), '')
+end
+
+assert 'Fixnum#==(Rational), Fixnum#!=(Rational)' do
+ assert_equal_rational(true, 2, Rational(4,2))
+ assert_equal_rational(true, -2, Rational(-4,2))
+ assert_equal_rational(true, -2, Rational(4,-2))
+ assert_equal_rational(false, 3, Rational(13,4))
+end
+
+assert 'Float#==(Rational), Float#!=(Rational)' do
+ assert_equal_rational(true, 2.0, Rational(4,2))
+ assert_equal_rational(true, -2.0, Rational(-4,2))
+ assert_equal_rational(true, -2.0, Rational(4,-2))
+ assert_equal_rational(false, 3.3, Rational(13,4))
+end