[前][次][番号順一覧][スレッド一覧]

ruby-changes:43629

From: mrkn <ko1@a...>
Date: Sun, 17 Jul 2016 23:53:06 +0900 (JST)
Subject: [ruby-changes:43629] mrkn:r55702 (trunk): numeric.c, complex.c: Add finite? and infinite? consistent with Float

mrkn	2016-07-17 23:53:00 +0900 (Sun, 17 Jul 2016)

  New Revision: 55702

  https://svn.ruby-lang.org/cgi-bin/viewvc.cgi?view=revision&revision=55702

  Log:
    numeric.c, complex.c: Add finite? and infinite? consistent with Float
    
    * numeric.c (num_finite_p, num_infinite_p): Add Numeric#finite? and
      Numeric#infinite?  [Feature #12039] [ruby-core:73618]
    
    * complex.c (rb_complex_finite_p): Add Complex#finite?
    
    * complex.c (rb_complex_infinite_p): Add Complex#infinite?
    
    * test/ruby/test_bignum.rb: Add test for Integer#finite? and
      Integer#infinite?
    
    * test/ruby/test_fixnum.rb: ditto.
    
    * test/ruby/test_rational.rb: Add test for Rational#finite? and
      Rational#infinite?
    
    * test/ruby/test_complex.rb: Add test for Complex#finite? and
      Complex#infinite?

  Modified files:
    trunk/ChangeLog
    trunk/complex.c
    trunk/numeric.c
    trunk/test/ruby/test_bignum.rb
    trunk/test/ruby/test_complex.rb
    trunk/test/ruby/test_fixnum.rb
    trunk/test/ruby/test_rational.rb
Index: complex.c
===================================================================
--- complex.c	(revision 55701)
+++ complex.c	(revision 55702)
@@ -1331,6 +1331,68 @@ nucomp_inspect(VALUE self) https://github.com/ruby/ruby/blob/trunk/complex.c#L1331
     return s;
 }
 
+/*
+ * call-seq:
+ *    cmp.finite?  ->  true or false
+ *
+ * Returns +true+ if +cmp+'s magnitude is finite number,
+ * oterwise returns +false+.
+ */
+static VALUE
+rb_complex_finite_p(VALUE self)
+{
+    VALUE magnitude = nucomp_abs(self);
+    double f;
+
+    switch (TYPE(magnitude)) {
+    case T_FIXNUM: case T_BIGNUM: case T_RATIONAL:
+	return Qtrue;
+
+    case T_FLOAT:
+	f = RFLOAT_VALUE(magnitude);
+	return isinf(f) ? Qfalse : Qtrue;
+
+    default:
+	return rb_funcall(magnitude, rb_intern("finite?"), 0);
+    }
+}
+
+/*
+ * call-seq:
+ *    cmp.infinite?  ->  nil or 1 or -1
+ *
+ * Returns values corresponding to the value of +cmp+'s magnitude:
+ *
+ * +finite+::    +nil+
+ * ++Infinity+:: ++1+
+ *
+ *  For example:
+ *
+ *     (1+1i).infinite?                   #=> nil
+ *     (Float::INFINITY + 1i).infinite?   #=> 1
+ */
+static VALUE
+rb_complex_infinite_p(VALUE self)
+{
+    VALUE magnitude = nucomp_abs(self);
+    double f;
+
+    switch (TYPE(magnitude)) {
+    case T_FIXNUM: case T_BIGNUM: case T_RATIONAL:
+	return Qnil;
+
+    case T_FLOAT:
+	f = RFLOAT_VALUE(magnitude);
+	if (isinf(f)) {
+	    return INT2FIX(f < 0 ? -1 : 1);
+	}
+	return Qnil;
+
+    default:
+	return rb_funcall(magnitude, rb_intern("infinite?"), 0);
+    }
+}
+
 /* :nodoc: */
 static VALUE
 nucomp_dumper(VALUE self)
@@ -2233,6 +2295,9 @@ Init_Complex(void) https://github.com/ruby/ruby/blob/trunk/complex.c#L2295
     rb_undef_method(rb_cComplex, "positive?");
     rb_undef_method(rb_cComplex, "negative?");
 
+    rb_define_method(rb_cComplex, "finite?", rb_complex_finite_p, 0);
+    rb_define_method(rb_cComplex, "infinite?", rb_complex_infinite_p, 0);
+
     rb_define_private_method(rb_cComplex, "marshal_dump", nucomp_marshal_dump, 0);
     compat = rb_define_class_under(rb_cComplex, "compatible", rb_cObject); /* :nodoc: */
     rb_define_private_method(compat, "marshal_load", nucomp_marshal_load, 1);
Index: ChangeLog
===================================================================
--- ChangeLog	(revision 55701)
+++ ChangeLog	(revision 55702)
@@ -1,3 +1,23 @@ https://github.com/ruby/ruby/blob/trunk/ChangeLog#L1
+Sun Jul 17 23:42:00 2016  Kenta Murata  <mrkn@m...>
+
+	* numeric.c (num_finite_p, num_infinite_p): Add Numeric#finite? and
+	  Numeric#infinite?  [Feature #12039] [ruby-core:73618]
+
+	* complex.c (rb_complex_finite_p): Add Complex#finite?
+
+	* complex.c (rb_complex_infinite_p): Add Complex#infinite?
+
+	* test/ruby/test_bignum.rb: Add test for Integer#finite? and
+	  Integer#infinite?
+
+	* test/ruby/test_fixnum.rb: ditto.
+
+	* test/ruby/test_rational.rb: Add test for Rational#finite? and
+	  Rational#infinite?
+
+	* test/ruby/test_complex.rb: Add test for Complex#finite? and
+	  Complex#infinite?
+
 Sun Jul 17 20:59:24 2016  Nobuyoshi Nakada  <nobu@r...>
 
 	* common.mk, enc/depend (casefold.h, name2ctype.h): move to
Index: test/ruby/test_rational.rb
===================================================================
--- test/ruby/test_rational.rb	(revision 55701)
+++ test/ruby/test_rational.rb	(revision 55702)
@@ -939,4 +939,13 @@ class Rational_Test < Test::Unit::TestCa https://github.com/ruby/ruby/blob/trunk/test/ruby/test_rational.rb#L939
   def test_known_bug
   end
 
+  def test_finite_p
+    assert_predicate(1/2r, :finite?)
+    assert_predicate(-1/2r, :finite?)
+  end
+
+  def test_infinite_p
+    assert_nil((1/2r).infinite?)
+    assert_nil((-1/2r).infinite?)
+  end
 end
Index: test/ruby/test_fixnum.rb
===================================================================
--- test/ruby/test_fixnum.rb	(revision 55701)
+++ test/ruby/test_fixnum.rb	(revision 55702)
@@ -337,4 +337,16 @@ class TestFixnum < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_fixnum.rb#L337
     assert_not_predicate(0, :negative?)
     assert_not_predicate(1, :negative?)
   end
+
+  def test_finite_p
+    assert_predicate(1, :finite?)
+    assert_predicate(0, :finite?)
+    assert_predicate(-1, :finite?)
+  end
+
+  def test_infinite_p
+    assert_nil(1.infinite?)
+    assert_nil(0.infinite?)
+    assert_nil(-1.infinite?)
+  end
 end
Index: test/ruby/test_complex.rb
===================================================================
--- test/ruby/test_complex.rb	(revision 55701)
+++ test/ruby/test_complex.rb	(revision 55702)
@@ -826,6 +826,30 @@ class Complex_Test < Test::Unit::TestCas https://github.com/ruby/ruby/blob/trunk/test/ruby/test_complex.rb#L826
     end
   end
 
+  def test_finite_p
+    assert_predicate(1+1i, :finite?)
+    assert_predicate(1-1i, :finite?)
+    assert_predicate(-1+1i, :finite?)
+    assert_predicate(-1-1i, :finite?)
+    assert_not_predicate(Float::INFINITY + 1i, :finite?)
+    assert_not_predicate(Complex(1, Float::INFINITY), :finite?)
+  end
+
+  def test_infinite_p
+    assert_nil((1+1i).infinite?)
+    assert_nil((1-1i).infinite?)
+    assert_nil((-1+1i).infinite?)
+    assert_nil((-1-1i).infinite?)
+    assert_equal(1, (Float::INFINITY + 1i).infinite?)
+    assert_equal(1, (Float::INFINITY - 1i).infinite?)
+    assert_equal(1, (-Float::INFINITY + 1i).infinite?)
+    assert_equal(1, (-Float::INFINITY - 1i).infinite?)
+    assert_equal(1, Complex(1, Float::INFINITY).infinite?)
+    assert_equal(1, Complex(-1, Float::INFINITY).infinite?)
+    assert_equal(1, Complex(1, -Float::INFINITY).infinite?)
+    assert_equal(1, Complex(-1, -Float::INFINITY).infinite?)
+  end
+
   def test_supp
     assert_equal(true, 1.real?)
     assert_equal(true, 1.1.real?)
Index: test/ruby/test_bignum.rb
===================================================================
--- test/ruby/test_bignum.rb	(revision 55701)
+++ test/ruby/test_bignum.rb	(revision 55702)
@@ -778,5 +778,15 @@ class TestBignum < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_bignum.rb#L778
     assert_raise(TypeError) { T1024P.digits("10") }
     assert_raise(TypeError) { T1024P.digits("a") }
   end
+
+  def test_finite_p
+    assert_predicate(T1024P, :finite?)
+    assert_predicate(-T1024P, :finite?)
+  end
+
+  def test_infinite_p
+    assert_nil(T1024P.infinite?)
+    assert_nil((-T1024P).infinite?)
+  end
 end
 end
Index: numeric.c
===================================================================
--- numeric.c	(revision 55701)
+++ numeric.c	(revision 55702)
@@ -673,6 +673,34 @@ num_nonzero_p(VALUE num) https://github.com/ruby/ruby/blob/trunk/numeric.c#L673
 
 /*
  *  call-seq:
+ *     num.finite?  ->  true or false
+ *
+ *  Return true if +num+ is finite number, oterwise returns false.
+ */
+static VALUE
+num_finite_p(VALUE num)
+{
+    return Qtrue;
+}
+
+/*
+ *  call-seq:
+ *     num.infinite?  ->  nil or 1 or -1
+ *
+ *  Returns values corresponding to the value of +num+'s magnitude:
+ *
+ *  +finite+::    +nil+
+ *  +-Infinity+:: +-1+
+ *  ++Infinity+:: ++1+
+ */
+static VALUE
+num_infinite_p(VALUE num)
+{
+    return Qnil;
+}
+
+/*
+ *  call-seq:
  *     num.to_int  ->  integer
  *
  *  Invokes the child class's +to_i+ method to convert +num+ to an integer.
@@ -5002,6 +5030,8 @@ Init_Numeric(void) https://github.com/ruby/ruby/blob/trunk/numeric.c#L5030
     rb_define_method(rb_cNumeric, "integer?", num_int_p, 0);
     rb_define_method(rb_cNumeric, "zero?", num_zero_p, 0);
     rb_define_method(rb_cNumeric, "nonzero?", num_nonzero_p, 0);
+    rb_define_method(rb_cNumeric, "finite?", num_finite_p, 0);
+    rb_define_method(rb_cNumeric, "infinite?", num_infinite_p, 0);
 
     rb_define_method(rb_cNumeric, "floor", num_floor, -1);
     rb_define_method(rb_cNumeric, "ceil", num_ceil, -1);

--
ML: ruby-changes@q...
Info: http://www.atdot.net/~ko1/quickml/

[前][次][番号順一覧][スレッド一覧]