| /* |
| * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. |
| * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
| * |
| * This code is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License version 2 only, as |
| * published by the Free Software Foundation. |
| * |
| * This code is distributed in the hope that it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
| * version 2 for more details (a copy is included in the LICENSE file that |
| * accompanied this code). |
| * |
| * You should have received a copy of the GNU General Public License version |
| * 2 along with this work; if not, write to the Free Software Foundation, |
| * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
| * |
| * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
| * or visit www.oracle.com if you need additional information or have any |
| * questions. |
| */ |
| package test.java.math.BigDecimal; |
| |
| /* |
| * @test |
| * @bug 4851776 4907265 6177836 6876282 8066842 |
| * @summary Some tests for the divide methods. |
| * @author Joseph D. Darcy |
| */ |
| |
| import java.math.*; |
| import static java.math.BigDecimal.*; |
| |
| import org.testng.Assert; |
| import org.testng.annotations.Test; |
| |
| // Android-changed: Replace error counting with asserts. |
| public class DivideTests { |
| |
| // Preliminary exact divide method; could be used for comparison |
| // purposes. |
| BigDecimal anotherDivide(BigDecimal dividend, BigDecimal divisor) { |
| /* |
| * Handle zero cases first. |
| */ |
| if (divisor.signum() == 0) { // x/0 |
| if (dividend.signum() == 0) // 0/0 |
| throw new ArithmeticException("Division undefined"); // NaN |
| throw new ArithmeticException("Division by zero"); |
| } |
| if (dividend.signum() == 0) // 0/y |
| return BigDecimal.ZERO; |
| else { |
| /* |
| * Determine if there is a result with a terminating |
| * decimal expansion. Putting aside overflow and |
| * underflow considerations, the existance of an exact |
| * result only depends on the ratio of the intVal's of the |
| * dividend (i.e. this) and and divisor since the scales |
| * of the argument just affect where the decimal point |
| * lies. |
| * |
| * For the ratio of (a = this.intVal) and (b = |
| * divisor.intVal) to have a finite decimal expansion, |
| * once a/b is put in lowest terms, b must be equal to |
| * (2^i)*(5^j) for some integer i,j >= 0. Therefore, we |
| * first compute to see if b_prime =(b/gcd(a,b)) is equal |
| * to (2^i)*(5^j). |
| */ |
| BigInteger TWO = BigInteger.valueOf(2); |
| BigInteger FIVE = BigInteger.valueOf(5); |
| BigInteger TEN = BigInteger.valueOf(10); |
| |
| BigInteger divisorIntvalue = divisor.scaleByPowerOfTen(divisor.scale()).toBigInteger().abs(); |
| BigInteger dividendIntvalue = dividend.scaleByPowerOfTen(dividend.scale()).toBigInteger().abs(); |
| |
| BigInteger b_prime = divisorIntvalue.divide(dividendIntvalue.gcd(divisorIntvalue)); |
| |
| boolean goodDivisor = false; |
| int i=0, j=0; |
| |
| badDivisor: { |
| while(! b_prime.equals(BigInteger.ONE) ) { |
| int b_primeModTen = b_prime.mod(TEN).intValue() ; |
| |
| switch(b_primeModTen) { |
| case 0: |
| // b_prime divisible by 10=2*5, increment i and j |
| i++; |
| j++; |
| b_prime = b_prime.divide(TEN); |
| break; |
| |
| case 5: |
| // b_prime divisible by 5, increment j |
| j++; |
| b_prime = b_prime.divide(FIVE); |
| break; |
| |
| case 2: |
| case 4: |
| case 6: |
| case 8: |
| // b_prime divisible by 2, increment i |
| i++; |
| b_prime = b_prime.divide(TWO); |
| break; |
| |
| default: // hit something we shouldn't have |
| b_prime = BigInteger.ONE; // terminate loop |
| break badDivisor; |
| } |
| } |
| |
| goodDivisor = true; |
| } |
| |
| if( ! goodDivisor ) { |
| throw new ArithmeticException("Non terminating decimal expansion"); |
| } |
| else { |
| // What is a rule for determining how many digits are |
| // needed? Once that is determined, cons up a new |
| // MathContext object and pass it on to the divide(bd, |
| // mc) method; precision == ?, roundingMode is unnecessary. |
| |
| // Are we sure this is the right scale to use? Should |
| // also determine a precision-based method. |
| MathContext mc = new MathContext(dividend.precision() + |
| (int)Math.ceil( |
| 10.0*divisor.precision()/3.0), |
| RoundingMode.UNNECESSARY); |
| // Should do some more work here to rescale, etc. |
| return dividend.divide(divisor, mc); |
| } |
| } |
| } |
| |
| @Test |
| public void powersOf2and5() { |
| for(int i = 0; i < 6; i++) { |
| int powerOf2 = (int)StrictMath.pow(2.0, i); |
| |
| for(int j = 0; j < 6; j++) { |
| int powerOf5 = (int)StrictMath.pow(5.0, j); |
| int product; |
| |
| BigDecimal bd; |
| |
| try { |
| bd = BigDecimal.ONE.divide(new BigDecimal(product=powerOf2*powerOf5)); |
| } catch (ArithmeticException e) { |
| Assert.fail((new BigDecimal(powerOf2)).toString() + " / " + |
| (new BigDecimal(powerOf5)).toString() + " threw an exception."); |
| } |
| |
| try { |
| bd = new BigDecimal(powerOf2).divide(new BigDecimal(powerOf5)); |
| } catch (ArithmeticException e) { |
| Assert.fail((new BigDecimal(powerOf2)).toString() + " / " + |
| (new BigDecimal(powerOf5)).toString() + " threw an exception."); |
| } |
| |
| try { |
| bd = new BigDecimal(powerOf5).divide(new BigDecimal(powerOf2)); |
| } catch (ArithmeticException e) { |
| Assert.fail((new BigDecimal(powerOf5)).toString() + " / " + |
| (new BigDecimal(powerOf2)).toString() + " threw an exception."); |
| } |
| |
| } |
| } |
| } |
| |
| @Test |
| public void nonTerminating() { |
| int[] primes = {1, 3, 7, 13, 17}; |
| |
| // For each pair of prime products, verify the ratio of |
| // non-equal products has a non-terminating expansion. |
| |
| for(int i = 0; i < primes.length; i++) { |
| for(int j = i+1; j < primes.length; j++) { |
| |
| for(int m = 0; m < primes.length; m++) { |
| for(int n = m+1; n < primes.length; n++) { |
| int dividend = primes[i] * primes[j]; |
| int divisor = primes[m] * primes[n]; |
| |
| if ( ((dividend/divisor) * divisor) != dividend ) { |
| try { |
| BigDecimal quotient = (new BigDecimal(dividend). |
| divide(new BigDecimal(divisor))); |
| Assert.fail("Exact quotient " + quotient.toString() + |
| " returned for non-terminating fraction " + |
| dividend + " / " + divisor + "."); |
| } |
| catch (ArithmeticException e) { |
| ; // Correct result |
| } |
| } |
| |
| } |
| } |
| } |
| } |
| } |
| |
| @Test |
| public void properScaleTests(){ |
| BigDecimal[][] testCases = { |
| {new BigDecimal("1"), new BigDecimal("5"), new BigDecimal("2e-1")}, |
| {new BigDecimal("1"), new BigDecimal("50e-1"), new BigDecimal("2e-1")}, |
| {new BigDecimal("10e-1"), new BigDecimal("5"), new BigDecimal("2e-1")}, |
| {new BigDecimal("1"), new BigDecimal("500e-2"), new BigDecimal("2e-1")}, |
| {new BigDecimal("100e-2"), new BigDecimal("5"), new BigDecimal("20e-2")}, |
| {new BigDecimal("1"), new BigDecimal("32"), new BigDecimal("3125e-5")}, |
| {new BigDecimal("1"), new BigDecimal("64"), new BigDecimal("15625e-6")}, |
| {new BigDecimal("1.0000000"), new BigDecimal("64"), new BigDecimal("156250e-7")}, |
| }; |
| |
| |
| for(BigDecimal[] tc : testCases) { |
| BigDecimal quotient = tc[0].divide(tc[1]); |
| Assert.assertEquals(quotient, tc[2], |
| "Unexpected quotient from " + tc[0] + " / " + tc[1] + |
| "; expected " + tc[2] + " got " + quotient); |
| } |
| } |
| |
| @Test |
| public void trailingZeroTests() { |
| MathContext mc = new MathContext(3, RoundingMode.FLOOR); |
| BigDecimal[][] testCases = { |
| {new BigDecimal("19"), new BigDecimal("100"), new BigDecimal("0.19")}, |
| {new BigDecimal("21"), new BigDecimal("110"), new BigDecimal("0.190")}, |
| }; |
| |
| for(BigDecimal[] tc : testCases) { |
| BigDecimal quotient = tc[0].divide(tc[1], mc); |
| Assert.assertEquals(quotient, tc[2], |
| "Unexpected quotient from " + tc[0] + " / " + tc[1] + |
| "; expected " + tc[2] + " got " + quotient); |
| } |
| } |
| |
| @Test |
| public void scaledRoundedDivideTests() { |
| // Tests of the traditional scaled divide under different |
| // rounding modes. |
| |
| // Encode rounding mode and scale for the divide in a |
| // BigDecimal with the significand equal to the rounding mode |
| // and the scale equal to the number's scale. |
| |
| // {dividend, divisor, rounding, quotient} |
| BigDecimal a = new BigDecimal("31415"); |
| BigDecimal a_minus = a.negate(); |
| BigDecimal b = new BigDecimal("10000"); |
| |
| BigDecimal c = new BigDecimal("31425"); |
| BigDecimal c_minus = c.negate(); |
| |
| // Ad hoc tests |
| BigDecimal d = new BigDecimal(new BigInteger("-37361671119238118911893939591735"), 10); |
| BigDecimal e = new BigDecimal(new BigInteger("74723342238476237823787879183470"), 15); |
| |
| BigDecimal[][] testCases = { |
| {a, b, BigDecimal.valueOf(ROUND_UP, 3), new BigDecimal("3.142")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_UP, 3), new BigDecimal("-3.142")}, |
| |
| {a, b, BigDecimal.valueOf(ROUND_DOWN, 3), new BigDecimal("3.141")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_DOWN, 3), new BigDecimal("-3.141")}, |
| |
| {a, b, BigDecimal.valueOf(ROUND_CEILING, 3), new BigDecimal("3.142")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_CEILING, 3), new BigDecimal("-3.141")}, |
| |
| {a, b, BigDecimal.valueOf(ROUND_FLOOR, 3), new BigDecimal("3.141")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_FLOOR, 3), new BigDecimal("-3.142")}, |
| |
| {a, b, BigDecimal.valueOf(ROUND_HALF_UP, 3), new BigDecimal("3.142")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_HALF_UP, 3), new BigDecimal("-3.142")}, |
| |
| {a, b, BigDecimal.valueOf(ROUND_DOWN, 3), new BigDecimal("3.141")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_DOWN, 3), new BigDecimal("-3.141")}, |
| |
| {a, b, BigDecimal.valueOf(ROUND_HALF_EVEN, 3), new BigDecimal("3.142")}, |
| {a_minus, b, BigDecimal.valueOf(ROUND_HALF_EVEN, 3), new BigDecimal("-3.142")}, |
| |
| {c, b, BigDecimal.valueOf(ROUND_HALF_EVEN, 3), new BigDecimal("3.142")}, |
| {c_minus, b, BigDecimal.valueOf(ROUND_HALF_EVEN, 3), new BigDecimal("-3.142")}, |
| |
| {d, e, BigDecimal.valueOf(ROUND_HALF_UP, -5), BigDecimal.valueOf(-1, -5)}, |
| {d, e, BigDecimal.valueOf(ROUND_HALF_DOWN, -5), BigDecimal.valueOf(0, -5)}, |
| {d, e, BigDecimal.valueOf(ROUND_HALF_EVEN, -5), BigDecimal.valueOf(0, -5)}, |
| }; |
| |
| for(BigDecimal[] tc : testCases) { |
| int scale = tc[2].scale(); |
| int rm = tc[2].unscaledValue().intValue(); |
| |
| BigDecimal quotient = tc[0].divide(tc[1], scale, rm); |
| Assert.assertEquals(quotient, tc[3], |
| "Unexpected quotient from " + tc[0] + " / " + tc[1] + |
| " scale " + scale + " rounding mode " + RoundingMode.valueOf(rm) + |
| "; expected " + tc[3] + " got " + quotient); |
| } |
| |
| // 6876282 |
| BigDecimal[][] testCases2 = { |
| // { dividend, divisor, expected quotient } |
| { new BigDecimal(3090), new BigDecimal(7), new BigDecimal(441) }, |
| { new BigDecimal("309000000000000000000000"), new BigDecimal("700000000000000000000"), |
| new BigDecimal(441) }, |
| { new BigDecimal("962.430000000000"), new BigDecimal("8346463.460000000000"), |
| new BigDecimal("0.000115309916") }, |
| { new BigDecimal("18446744073709551631"), new BigDecimal("4611686018427387909"), |
| new BigDecimal(4) }, |
| { new BigDecimal("18446744073709551630"), new BigDecimal("4611686018427387909"), |
| new BigDecimal(4) }, |
| { new BigDecimal("23058430092136939523"), new BigDecimal("4611686018427387905"), |
| new BigDecimal(5) }, |
| { new BigDecimal("-18446744073709551661"), new BigDecimal("-4611686018427387919"), |
| new BigDecimal(4) }, |
| { new BigDecimal("-18446744073709551660"), new BigDecimal("-4611686018427387919"), |
| new BigDecimal(4) }, |
| }; |
| |
| for (BigDecimal[] test : testCases2) { |
| BigDecimal quo = test[0].divide(test[1], RoundingMode.HALF_UP); |
| Assert.assertEquals(quo, test[2], "Unexpected quotient from " + test[0] + " / " + test[1] + |
| " rounding mode HALF_UP" + |
| "; expected " + test[2] + " got " + quo); |
| } |
| } |
| |
| @Test |
| public void divideByOneTests() { |
| //problematic divisor: one with scale 17 |
| BigDecimal one = BigDecimal.ONE.setScale(17); |
| RoundingMode rounding = RoundingMode.UNNECESSARY; |
| |
| long[][] unscaledAndScale = new long[][] { |
| { Long.MAX_VALUE, 17}, |
| {-Long.MAX_VALUE, 17}, |
| { Long.MAX_VALUE, 0}, |
| {-Long.MAX_VALUE, 0}, |
| { Long.MAX_VALUE, 100}, |
| {-Long.MAX_VALUE, 100} |
| }; |
| |
| for (long[] uas : unscaledAndScale) { |
| long unscaled = uas[0]; |
| int scale = (int)uas[1]; |
| |
| BigDecimal noRound = null; |
| try { |
| noRound = BigDecimal.valueOf(unscaled, scale). |
| divide(one, RoundingMode.UNNECESSARY); |
| } catch (ArithmeticException e) { |
| Assert.fail("ArithmeticException for value " + unscaled |
| + " and scale " + scale + " without rounding"); |
| } |
| |
| BigDecimal roundDown = null; |
| try { |
| roundDown = BigDecimal.valueOf(unscaled, scale). |
| divide(one, RoundingMode.DOWN); |
| } catch (ArithmeticException e) { |
| Assert.fail("ArithmeticException for value " + unscaled |
| + " and scale " + scale + " with rounding down"); |
| } |
| |
| if (noRound != null && roundDown != null && noRound.compareTo(roundDown) != 0) { |
| Assert.fail("Equality failure for value " + unscaled |
| + " and scale " + scale); |
| } |
| } |
| } |
| } |