blob: cca400137115c690c019c823b8b2578d6b00e509 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.commons.numbers.core;
import java.math.BigDecimal;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
/**
* Tests for examples contained in the user guide.
*/
class UserGuideTest {
@Test
void testNorm1() {
double x = Norm.EUCLIDEAN.of(3, -4);
double y = Norm.MANHATTAN.of(3, -4, 5);
double z = Norm.MAXIMUM.of(new double[] {3, -4, 5, -6, -7, -8});
Assertions.assertEquals(5, x);
Assertions.assertEquals(12, y);
Assertions.assertEquals(8, z);
double big = Double.MAX_VALUE * 0.5;
double length = Norm.EUCLIDEAN.of(big, big, big);
double expected = Math.sqrt(0.5 * 0.5 * 3) * Double.MAX_VALUE;
Assertions.assertEquals(expected, length, Math.ulp(expected));
}
@Test
void testSum1() {
double sum1 = Sum.create().add(1)
.addProduct(3, 4)
.getAsDouble();
double sum2 = Sum.of(1).addProduct(3, 4)
.getAsDouble();
double sum3 = Sum.ofProducts(new double[] {3, 4}, new double[] {5, 6})
.getAsDouble();
Assertions.assertEquals(13, sum1);
Assertions.assertEquals(13, sum2);
Assertions.assertEquals(3 * 5 + 4 * 6, sum3);
Assertions.assertEquals(Double.NaN, Sum.of(1, 2, Double.NaN).getAsDouble());
Assertions.assertEquals(Double.NEGATIVE_INFINITY, Sum.of(1, 2, Double.NEGATIVE_INFINITY).getAsDouble());
}
@Test
void testSum2() {
double x1 = 1e100 + 1 - 2 - 1e100;
double x2 = Sum.of(1e100, 1, -2, -1e100).getAsDouble();
Assertions.assertEquals(0.0, x1);
Assertions.assertEquals(-1.0, x2);
}
@Test
void testSum3() {
double x1 = 1e100 + 1 - 2 - 1e100;
Sum s1 = Sum.of(1e100, 1);
Sum s2 = Sum.of(2, 1e100);
double x2 = s1.subtract(s2).getAsDouble();
Assertions.assertEquals(0.0, x1);
Assertions.assertEquals(-1.0, x2);
}
@Test
void testPrecision1() {
// Default allows no numbers between
Assertions.assertTrue(Precision.equals(1000.0, 1000.0));
Assertions.assertTrue(Precision.equals(1000.0, 1000.0 + Math.ulp(1000.0)));
Assertions.assertFalse(Precision.equals(1000.0, 1000.0 + 2 * Math.ulp(1000.0)));
// Absolute - tolerance is floating-point
Assertions.assertFalse(Precision.equals(1000.0, 1001.0));
Assertions.assertTrue(Precision.equals(1000.0, 1001.0, 1.0));
Assertions.assertTrue(Precision.equals(1000.0, 1000.0 + Math.ulp(1000.0), 0.0));
// ULP - tolerance is integer
Assertions.assertFalse(Precision.equals(1000.0, 1001.0));
Assertions.assertFalse(Precision.equals(1000.0, 1001.0, 1));
Assertions.assertFalse(Precision.equals(1000.0, 1000.0 + 2 * Math.ulp(1000.0), 1));
Assertions.assertTrue(Precision.equals(1000.0, 1000.0 + 2 * Math.ulp(1000.0), 2));
Assertions.assertFalse(Precision.equals(1000.0, 1000.0 + 3 * Math.ulp(1000.0), 2));
// Relative
Assertions.assertFalse(Precision.equalsWithRelativeTolerance(1000.0, 1001.0, 1e-6));
Assertions.assertTrue(Precision.equalsWithRelativeTolerance(1000.0, 1001.0, 1e-3));
}
@Test
void testPrecision2() {
Assertions.assertFalse(Precision.equals(Double.NaN, 1000.0));
Assertions.assertFalse(Precision.equals(Double.NaN, Double.NaN));
Assertions.assertTrue(Precision.equalsIncludingNaN(Double.NaN, Double.NaN));
Assertions.assertTrue(Precision.equals(Double.POSITIVE_INFINITY, Double.POSITIVE_INFINITY));
Assertions.assertTrue(Precision.equals(Double.NEGATIVE_INFINITY, Double.NEGATIVE_INFINITY));
Assertions.assertFalse(Precision.equals(Double.POSITIVE_INFINITY, Double.NEGATIVE_INFINITY));
}
@Test
void testPrecision3() {
Assertions.assertEquals(0, Precision.compareTo(100, 100, 0.0));
Assertions.assertEquals(0, Precision.compareTo(100, 101, 1.0));
Assertions.assertEquals(-1, Precision.compareTo(100, 102, 1.0));
Assertions.assertEquals(1, Precision.compareTo(102, 100, 1.0));
}
@Test
void testPrecision4() {
Precision.DoubleEquivalence eq = Precision.doubleEquivalenceOfEpsilon(1.0);
Assertions.assertFalse(eq.lt(100, 100));
Assertions.assertTrue(eq.lte(100, 100));
Assertions.assertTrue(eq.eq(100, 100));
Assertions.assertTrue(eq.gte(100, 100));
Assertions.assertFalse(eq.gt(100, 100));
}
@Test
void testPrecision5() {
Assertions.assertEquals(678.125, Precision.round(678.125, 4));
Assertions.assertEquals(678.125, Precision.round(678.125, 3));
Assertions.assertEquals(678.13, Precision.round(678.125, 2));
Assertions.assertEquals(678.1, Precision.round(678.125, 1));
Assertions.assertEquals(678.0, Precision.round(678.125, 0));
Assertions.assertEquals(680.0, Precision.round(678.125, -1));
Assertions.assertEquals(700.0, Precision.round(678.125, -2));
Assertions.assertEquals(0.10000000000000009, Precision.representableDelta(1.0, 0.1));
}
@Test
void testDD1() {
double x = Math.PI;
int y = 42;
long z = -8564728970587006436L;
Assertions.assertEquals(x, DD.of(x).doubleValue());
Assertions.assertEquals(y, DD.of(y).intValue());
Assertions.assertEquals(z, DD.of(z).longValue());
Assertions.assertNotEquals(z, (long) (double) z);
}
@Test
void testDD2() {
BigDecimal pi = new BigDecimal("3.14159265358979323846264338327950288419716939937510");
DD x = DD.from(pi);
Assertions.assertEquals("(3.141592653589793,1.2246467991473532E-16)", x.toString());
Assertions.assertNotEquals(0, pi.compareTo(x.bigDecimalValue()));
Assertions.assertEquals(Math.PI, x.hi());
Assertions.assertEquals(pi.subtract(new BigDecimal(Math.PI)).doubleValue(), x.lo());
DD nan = DD.of(Double.NaN);
Assertions.assertFalse(nan.isFinite());
Assertions.assertThrows(NumberFormatException.class, () -> nan.bigDecimalValue());
}
@Test
void testDD3() {
long x = -8564728970587006436L;
Assertions.assertNotEquals(x + 1, DD.ONE.add(x).longValue());
Assertions.assertEquals(x + 1, DD.ONE.add(DD.of(x)).longValue());
}
@Test
void testDD4() {
double a = 1.2345678901234567;
double b = 123.45678901234567;
DD w = DD.ofProduct(a, b);
DD x = DD.ofSum(a, b);
DD y = DD.ofDifference(a, b);
DD z = DD.fromQuotient(1, 3);
Assertions.assertEquals("(152.41578753238835,-1.0325951435749745E-14)", w.toString());
Assertions.assertEquals("(124.69135690246912,-1.1102230246251565E-15)", x.toString());
Assertions.assertEquals("(-122.22222112222221,-1.1102230246251565E-15)", y.toString());
Assertions.assertEquals("(0.3333333333333333,1.850371707708594E-17)", z.toString());
Assertions.assertEquals(a * b, w.hi());
Assertions.assertEquals(a + b, x.hi());
Assertions.assertEquals(a - b, y.hi());
Assertions.assertEquals(1.0 / 3, z.hi());
DD zz = DD.of(1).divide(DD.of(3));
Assertions.assertEquals(z, zz);
}
@Test
void testDD5() {
Assertions.assertEquals(0.9999999999999999, 1.0 / 2 + 1.0 / 3 + 1.0 / 6);
DD z = DD.fromQuotient(1, 2)
.add(DD.fromQuotient(1, 3))
.add(DD.fromQuotient(1, 6));
Assertions.assertEquals("(1.0,-4.622231866529366E-33)", z.toString());
Assertions.assertEquals(1.0, z.doubleValue());
}
@Test
void testDD6() {
double a = 1;
double b = Math.pow(2, 53);
double c = Math.pow(2, 106);
DD z = DD.of(a).add(b).add(c).subtract(c).subtract(b);
Assertions.assertEquals(0.0, z.doubleValue());
}
@Test
void testDD7() {
double a = 1.5 * Math.pow(2, 1023);
double b = 4 * Math.pow(2, -1022);
DD x = DD.of(a);
DD y = DD.of(b);
Assertions.assertFalse(x.multiply(y).isFinite());
// Create fractional representation as [0.5, 1) * 2^b
int[] xb = {0};
int[] yb = {0};
x = x.frexp(xb); // (0.75, 0) * 2^1024
y = y.frexp(yb); // (0.5, 0) * 2^-1019
Assertions.assertEquals(0.75, x.doubleValue());
Assertions.assertEquals(0.5, y.doubleValue());
Assertions.assertEquals(1024, xb[0]);
Assertions.assertEquals(-1019, yb[0]);
DD z = x.multiply(y); // (0.375, 0)
Assertions.assertEquals(0.375, z.doubleValue());
// Rescale by 2^5
Assertions.assertEquals(a * b, z.scalb(xb[0] + yb[0]).doubleValue());
}
}