2016-05-27 14:33:48 +09:00
|
|
|
/*
|
2024-03-05 19:32:29 +00:00
|
|
|
* Copyright (c) 2003, 2024, Oracle and/or its affiliates. All rights reserved.
|
2016-05-27 14:33:48 +09:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
2024-06-26 17:10:09 +00:00
|
|
|
* @bug 4018937 8008577 8174269 8333755
|
2016-05-27 14:33:48 +09:00
|
|
|
* @summary Confirm that methods which are newly added to support BigDecimal and BigInteger work as expected.
|
2024-03-05 19:32:29 +00:00
|
|
|
* @run junit/othervm BigDecimalParse
|
2016-05-27 14:33:48 +09:00
|
|
|
*/
|
|
|
|
|
|
|
|
import java.math.BigDecimal;
|
2024-06-26 17:10:09 +00:00
|
|
|
import java.text.DecimalFormat;
|
|
|
|
import java.text.Format;
|
|
|
|
import java.text.MessageFormat;
|
|
|
|
import java.text.NumberFormat;
|
|
|
|
import java.text.ParsePosition;
|
|
|
|
import java.util.Locale;
|
2016-05-27 14:33:48 +09:00
|
|
|
|
2023-10-03 16:38:17 +00:00
|
|
|
import org.junit.jupiter.api.Test;
|
|
|
|
import org.junit.jupiter.api.BeforeAll;
|
2016-05-27 14:33:48 +09:00
|
|
|
|
2023-10-03 16:38:17 +00:00
|
|
|
import static org.junit.jupiter.api.Assertions.fail;
|
|
|
|
|
|
|
|
public class BigDecimalParse {
|
|
|
|
|
|
|
|
// Change JVM default Locale
|
|
|
|
@BeforeAll
|
|
|
|
static void initAll() {
|
2024-03-05 19:32:29 +00:00
|
|
|
Locale.setDefault(Locale.forLanguageTag("en-US-u-cf-account"));
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
2023-10-03 16:38:17 +00:00
|
|
|
|
2016-05-27 14:33:48 +09:00
|
|
|
static final String nonsep_int =
|
|
|
|
"123456789012345678901234567890123456789012345678901234567890" +
|
|
|
|
"123456789012345678901234567890123456789012345678901234567890" +
|
|
|
|
"123456789012345678901234567890123456789012345678901234567890" +
|
|
|
|
"123456789012345678901234567890123456789012345678901234567890" +
|
|
|
|
"123456789012345678901234567890123456789012345678901234567890" +
|
|
|
|
"123456789012345678901234567890123456789012345678901234567890";
|
|
|
|
|
|
|
|
static final String sep_int =
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890," +
|
|
|
|
"123,456,789,012,345,678,901,234,567,890";
|
|
|
|
|
|
|
|
static final String nonsep_zero =
|
|
|
|
"000000000000000000000000000000000000000000000000000000000000" +
|
|
|
|
"000000000000000000000000000000000000000000000000000000000000" +
|
|
|
|
"000000000000000000000000000000000000000000000000000000000000" +
|
|
|
|
"000000000000000000000000000000000000000000000000000000000000" +
|
|
|
|
"000000000000000000000000000000000000000000000000000000000000" +
|
|
|
|
"000000000000000000000000000000000000000000000000000000000000";
|
|
|
|
|
|
|
|
static final String sep_zero =
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000," +
|
|
|
|
"000,000,000,000,000,000,000,000,000,000";
|
|
|
|
|
|
|
|
static final String fra =
|
|
|
|
"012345678901234567890123456789012345678901234567890123456789" +
|
|
|
|
"012345678901234567890123456789012345678901234567890123456789" +
|
|
|
|
"012345678901234567890123456789012345678901234567890123456789" +
|
|
|
|
"012345678901234567890123456789012345678901234567890123456789" +
|
|
|
|
"012345678901234567890123456789012345678901234567890123456789" +
|
|
|
|
"012345678901234567890123456789012345678901234567890123456789";
|
|
|
|
|
|
|
|
|
|
|
|
Number parsed = null;
|
|
|
|
ParsePosition pp;
|
|
|
|
boolean exceptionOccurred;
|
|
|
|
String msg;
|
|
|
|
DecimalFormat df;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for normal big numbers which have the fraction part
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_BigDecimal() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
// From: 1234...7890.012...789
|
|
|
|
// To: BigDecimal 1234...7890.012...789
|
|
|
|
check(nonsep_int + "." + fra, new BigDecimal(nonsep_int + "." + fra));
|
|
|
|
|
|
|
|
// From: -1,234...7,890.012...789
|
|
|
|
// To: BigDecimal -1234...7890.012...789
|
|
|
|
check("-" + sep_int + "." + fra,
|
|
|
|
new BigDecimal("-" + nonsep_int + "." + fra));
|
|
|
|
|
|
|
|
// From: 000...0000.0...0
|
|
|
|
// To: BigDecimal 0E-360
|
|
|
|
check(nonsep_zero + "." + nonsep_zero,
|
|
|
|
new BigDecimal(nonsep_zero + "." + nonsep_zero));
|
|
|
|
|
|
|
|
// From: 0.000...0000123...789E370
|
|
|
|
// To: BigDecimal 0.0123...789
|
|
|
|
check("0.0000000000" + nonsep_zero + fra + "E370",
|
|
|
|
new BigDecimal("0.0000000000" + nonsep_zero + fra + "E370"));
|
|
|
|
|
|
|
|
// From: 0.1123...890E-360
|
|
|
|
// To: BigDecimal 1.123...890E-361
|
|
|
|
check("0.1" + nonsep_int + "E-360",
|
|
|
|
new BigDecimal("0.1" + nonsep_int + "E-360"));
|
|
|
|
|
|
|
|
// From: 000...0000.0...0123...7890
|
|
|
|
// To: BigDecimal 1.234...890E-361
|
|
|
|
check(nonsep_zero + "." + nonsep_zero + nonsep_int,
|
|
|
|
new BigDecimal(nonsep_zero + "." + nonsep_zero + nonsep_int));
|
|
|
|
|
|
|
|
// From: 0.123...890E360
|
|
|
|
// To: BigDecimal 123...890
|
|
|
|
check("0." + nonsep_int + "E360",
|
|
|
|
new BigDecimal("0." + nonsep_int + "E360"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for normal big numbers which have the fraction part with multiplier
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_BigDecimal_usingMultiplier() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
// From: 250,0...0,000.000...000
|
|
|
|
// To: 1000...0000.000...000
|
|
|
|
df.setMultiplier(250000000);
|
|
|
|
check("250,000,000," + sep_zero + "." + nonsep_zero,
|
|
|
|
new BigDecimal("1" + nonsep_zero + "." + nonsep_zero));
|
|
|
|
|
|
|
|
// From: -250,0...0,000.000...000
|
|
|
|
// To: -1000...0000.000...000
|
|
|
|
check("-250,000,000," + sep_zero + "." + nonsep_zero,
|
|
|
|
new BigDecimal("-1" + nonsep_zero + "." + nonsep_zero));
|
|
|
|
|
|
|
|
// From: 250,0...0,000.000...000
|
|
|
|
// To: -1000...0000.000...000
|
|
|
|
df.setMultiplier(-250000000);
|
|
|
|
check("250,000,000," + sep_zero + "." + nonsep_zero,
|
|
|
|
new BigDecimal("-1" + nonsep_zero + "." + nonsep_zero));
|
|
|
|
|
|
|
|
// From: -250,0...0,000.000...000
|
|
|
|
// To: 1000...0000.000...000
|
|
|
|
check("-250,000,000," + sep_zero + "." + nonsep_zero,
|
|
|
|
new BigDecimal("1" + nonsep_zero + "." + nonsep_zero));
|
|
|
|
|
|
|
|
// Confirm that ArithmeticException is handled properly
|
|
|
|
// From: 1000.000
|
|
|
|
// To: 333.333
|
|
|
|
df.setMultiplier(3);
|
|
|
|
check("1000.000", new BigDecimal("333.333"));
|
|
|
|
|
|
|
|
// Confirm that ArithmeticException is handled properly
|
|
|
|
// From: 10000.0000
|
|
|
|
// To: 303.0303
|
|
|
|
df.setMultiplier(33);
|
|
|
|
check("10000.0000", new BigDecimal("303.0303"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for division by zero (BigDecimal)
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_BigDecimal_DivisionByZero() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
df.setMultiplier(0);
|
|
|
|
|
|
|
|
// From: 1000.000
|
|
|
|
// To: Double.POSITIVE_INFINITY
|
2016-09-15 08:18:57 +09:00
|
|
|
check("1000.000", Double.POSITIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: -1000
|
|
|
|
// To: Double.NEGATIVE_INFINITY
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-1000", Double.NEGATIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: -0.00
|
|
|
|
// To: Double.NaN
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-0.00", Double.NaN);
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for division by zero (Double)
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_Double_DivisionByZero() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(false);
|
|
|
|
df.setMultiplier(0);
|
|
|
|
|
|
|
|
// From: 1000.000
|
|
|
|
// To: Double.POSITIVE_INFINITY
|
2016-09-15 08:18:57 +09:00
|
|
|
check("1000.000", Double.POSITIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: -1000.000
|
|
|
|
// To: Double.NEGATIVE_INFINITY
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-1000.000", Double.NEGATIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: 0.0
|
|
|
|
// To: Double.NaN
|
2016-09-15 08:18:57 +09:00
|
|
|
check("0.0", Double.NaN);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: -0.0 (Double)
|
|
|
|
// To: Double.NaN
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-0.0", Double.NaN);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: Double.NaN
|
|
|
|
// To: Double.NaN
|
2024-03-05 19:32:29 +00:00
|
|
|
check("NaN", Double.NaN);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: Double.POSITIVE_INFINITY
|
|
|
|
// To: Double.NaN
|
2016-09-15 08:18:57 +09:00
|
|
|
check("\u221e", Double.POSITIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: Double.NEGATIVE_INFINITY
|
|
|
|
// To: Double.NaN
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-\u221e", Double.NEGATIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for division by zero (Long)
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_Long_DivisionByZero() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(false);
|
|
|
|
df.setMultiplier(0);
|
|
|
|
|
|
|
|
// From: 1000
|
|
|
|
// To: Double.POSITIVE_INFINITY
|
2016-09-15 08:18:57 +09:00
|
|
|
check("1000", Double.POSITIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: -1000
|
|
|
|
// To: Double.NEGATIVE_INFINITY
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-1000", Double.NEGATIVE_INFINITY);
|
2016-05-27 14:33:48 +09:00
|
|
|
|
|
|
|
// From: -000 (Long)
|
|
|
|
// To: Double.NaN
|
2016-09-15 08:18:57 +09:00
|
|
|
check("-000", Double.NaN);
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for normal big numbers which don't have the fraction part
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_BigInteger() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
// From: 123...890
|
|
|
|
// To: BigDecimal 123...890
|
|
|
|
check(nonsep_int + nonsep_int, new BigDecimal(nonsep_int + nonsep_int));
|
|
|
|
|
|
|
|
// From: 123,4...7,890
|
|
|
|
// To: BigDecimal 1234...7890
|
|
|
|
check(sep_int + "," + sep_int, new BigDecimal(nonsep_int + nonsep_int));
|
|
|
|
|
|
|
|
// From: -000...000123...890
|
|
|
|
// To: BigDecimal -123...890
|
|
|
|
check("-" + nonsep_zero + nonsep_int, new BigDecimal("-" + nonsep_int));
|
|
|
|
|
|
|
|
// From: -000,0...0,000,123,4...7,890
|
|
|
|
// To: BigDecimal -123...890
|
|
|
|
check("-" + sep_zero + "," + sep_int, new BigDecimal("-" + nonsep_int));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for normal big numbers which don't have the fraction part with
|
|
|
|
* multiplier
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_BigInteger_usingMultiplier() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
// From: 250,0...0,000
|
|
|
|
// To: 1000...0000
|
|
|
|
df.setMultiplier(250000000);
|
|
|
|
check("250,000,000," + sep_zero, new BigDecimal("1" + nonsep_zero));
|
|
|
|
|
|
|
|
// From: -250,0...0,000
|
|
|
|
// To: -1000...0000
|
|
|
|
check("-250,000,000," + sep_zero, new BigDecimal("-1" + nonsep_zero));
|
|
|
|
|
|
|
|
// From: 250,0...0,000
|
|
|
|
// To: -1000...0000
|
|
|
|
df.setMultiplier(-250000000);
|
|
|
|
check("250,000,000," + sep_zero, new BigDecimal("-1" + nonsep_zero));
|
|
|
|
|
|
|
|
// From: -250,0...0,000
|
|
|
|
// To: 1000...0000
|
|
|
|
check("-250,000,000," + sep_zero, new BigDecimal("1" + nonsep_zero));
|
|
|
|
|
|
|
|
// From: 250,0...0,000E-360
|
|
|
|
// To: -1000...0000.000...000
|
|
|
|
check("250,000,000," + sep_zero + "," + sep_zero + "E-360",
|
|
|
|
new BigDecimal("-1" + nonsep_zero + "." + nonsep_zero));
|
|
|
|
|
|
|
|
// Confirm that a division which results in a irrational number is done
|
|
|
|
// properly
|
|
|
|
// From: 1000
|
|
|
|
// To: 333
|
|
|
|
df.setMultiplier(3);
|
|
|
|
check("1000", new BigDecimal("333"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for special numbers
|
|
|
|
* Double.NaN
|
|
|
|
* Double.POSITIVE_INFINITY
|
|
|
|
* Double.NEGATIVE_INFINITY
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_SpecialNumber() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
String[] numbers = {
|
2024-03-05 19:32:29 +00:00
|
|
|
"0", "0.0", "25", "25.0", "25.5", "\u221e", "NaN",
|
2016-05-27 14:33:48 +09:00
|
|
|
"-0", "-0.0", "-25", "-25.0", "-25.5", "-\u221e",
|
|
|
|
};
|
|
|
|
int multipliers[] = {5, -5};
|
|
|
|
Number[][] expected = {
|
|
|
|
{
|
|
|
|
new BigDecimal("0"), new BigDecimal("0.0"), new BigDecimal("5"),
|
|
|
|
new BigDecimal("5.0"), new BigDecimal("5.1"),
|
2016-09-15 08:18:57 +09:00
|
|
|
Double.POSITIVE_INFINITY, Double.NaN,
|
2016-05-27 14:33:48 +09:00
|
|
|
new BigDecimal("0"), new BigDecimal("0.0"),
|
|
|
|
new BigDecimal("-5"), new BigDecimal("-5.0"),
|
|
|
|
new BigDecimal("-5.1"),
|
2016-09-15 08:18:57 +09:00
|
|
|
Double.NEGATIVE_INFINITY, Double.NaN,
|
2016-05-27 14:33:48 +09:00
|
|
|
},
|
|
|
|
{
|
|
|
|
new BigDecimal("0"), new BigDecimal("0.0"),
|
|
|
|
new BigDecimal("-5"), new BigDecimal("-5.0"),
|
|
|
|
new BigDecimal("-5.1"),
|
2016-09-15 08:18:57 +09:00
|
|
|
Double.NEGATIVE_INFINITY, Double.NaN,
|
2016-05-27 14:33:48 +09:00
|
|
|
new BigDecimal("0"), new BigDecimal("0.0"), new BigDecimal("5"),
|
|
|
|
new BigDecimal("5.0"), new BigDecimal("5.1"),
|
2016-09-15 08:18:57 +09:00
|
|
|
Double.POSITIVE_INFINITY,
|
2016-05-27 14:33:48 +09:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; i < multipliers.length; i++) {
|
|
|
|
df.setMultiplier(multipliers[i]);
|
|
|
|
for (int j = 0; j < numbers.length; j++) {
|
|
|
|
check(String.valueOf(numbers[j]), expected[i][j]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for special numbers
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_Other() {
|
2016-05-27 14:33:48 +09:00
|
|
|
df = new DecimalFormat();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
String[] numbers = {
|
|
|
|
"-9223372036854775808", // Long.MIN_VALUE
|
|
|
|
};
|
|
|
|
int multipliers[] = {1, -1};
|
|
|
|
String[][] expected = {
|
|
|
|
{"-9223372036854775808"}, // Long.MIN_VALUE
|
|
|
|
{"9223372036854775808"}, // Long.MAX_VALUE+1 = abs(MIN_VALUE)
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; i < multipliers.length; i++) {
|
|
|
|
df.setMultiplier(multipliers[i]);
|
|
|
|
for (int j = 0; j < numbers.length; j++) {
|
|
|
|
check(String.valueOf(numbers[j]),
|
|
|
|
new BigDecimal(expected[i][j]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static final String[] patterns = {
|
|
|
|
" {0, number} ",
|
|
|
|
" {0, number} ",
|
|
|
|
" {0, number, currency} ",
|
|
|
|
" {0, number, currency} ",
|
|
|
|
" {0, number, percent} ",
|
|
|
|
" {0, number, percent} ",
|
|
|
|
" {0, number,#,##0.###E0} ",
|
|
|
|
" {0, number,#,##0.###E0} ",
|
|
|
|
|
|
|
|
" {0, number} ",
|
|
|
|
" {0, number} ",
|
|
|
|
" {0, number, integer} ",
|
|
|
|
" {0, number, integer} ",
|
|
|
|
" {0, number, currency} ",
|
|
|
|
" {0, number, currency} ",
|
|
|
|
" {0, number, percent} ",
|
|
|
|
" {0, number, percent} ",
|
|
|
|
" {0, number,#,##0.###E0} ",
|
|
|
|
" {0, number,#,##0.###E0} ",
|
|
|
|
};
|
|
|
|
static final String[] from = {
|
|
|
|
" 12,345,678,901,234,567,890.98765432109876543210987654321 ",
|
|
|
|
" -12,345,678,901,234,567,890.98765432109876543210987654321 ",
|
|
|
|
" $12,345,678,901,234,567,890.98765432109876543210987654321 ",
|
|
|
|
" ($12,345,678,901,234,567,890.98765432109876543210987654321) ",
|
|
|
|
" 1,234,567,890,123,456,789,098.76543210987654321098765432100% ",
|
|
|
|
" -1,234,567,890,123,456,789,098.76543210987654321098765432100% ",
|
|
|
|
" 12,345,678,901,234,567,890.98765432109876543210987654321E-20 ",
|
|
|
|
" -12,345,678,901,234,567,890.98765432109876543210987654321E-20 ",
|
|
|
|
|
|
|
|
" 9,876,543,210,987,654,321,098,765,432,109,876,543,210 ",
|
|
|
|
" -9,876,543,210,987,654,321,098,765,432,109,876,543,210 ",
|
|
|
|
" 9,876,543,210,987,654,321,098,765,432,109,876,543,210E5 ",
|
|
|
|
" -9,876,543,210,987,654,321,098,765,432,109,876,543,210E-5 ",
|
|
|
|
" $9,876,543,210,987,654,321,098,765,432,109,876,543,210.00 ",
|
|
|
|
" ($9,876,543,210,987,654,321,098,765,432,109,876,543,210.00) ",
|
|
|
|
" 987,654,321,098,765,432,109,876,543,210,987,654,321,012% ",
|
|
|
|
" -987,654,321,098,765,432,109,876,543,210,987,654,321,012% ",
|
|
|
|
" 98,765,432,109,876,543,210.98765432109876543210E20 ",
|
|
|
|
" -987,654,321,098,765,432,109,876,543,210,987,654,321,000,000,000,000,000,000,000E-20 ",
|
|
|
|
};
|
|
|
|
|
|
|
|
static final String[] expected1 = { // isParseIntegerOnly() == false
|
|
|
|
"12345678901234567890.98765432109876543210987654321",
|
|
|
|
"-12345678901234567890.98765432109876543210987654321",
|
|
|
|
"12345678901234567890.98765432109876543210987654321",
|
|
|
|
"-12345678901234567890.98765432109876543210987654321",
|
|
|
|
"12345678901234567890.98765432109876543210987654321",
|
|
|
|
"-12345678901234567890.98765432109876543210987654321",
|
|
|
|
"0.1234567890123456789098765432109876543210987654321",
|
|
|
|
"-0.1234567890123456789098765432109876543210987654321",
|
|
|
|
|
|
|
|
"9876543210987654321098765432109876543210",
|
|
|
|
"-9876543210987654321098765432109876543210",
|
|
|
|
"9.876543210987654321098765432109876543210E44",
|
|
|
|
"-98765432109876543210987654321098765.43210",
|
|
|
|
"9876543210987654321098765432109876543210.00",
|
|
|
|
"-9876543210987654321098765432109876543210.00",
|
|
|
|
"9876543210987654321098765432109876543210.12",
|
|
|
|
"-9876543210987654321098765432109876543210.12",
|
|
|
|
"9876543210987654321098765432109876543210",
|
|
|
|
"-9876543210987654321098765432109876543210.00000000000000000000",
|
|
|
|
};
|
|
|
|
static final int[] parsePosition1 = {
|
|
|
|
60, 61, 61, 63, 64, 65, 64, 65,
|
|
|
|
57, 58, 59, 61, 61, 63, 60, 61, 54, 88,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for MessageFormat: setParseIntegerOnly(false)
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_MessageFormat_NotParseIntegerOnly() {
|
2016-05-27 14:33:48 +09:00
|
|
|
for (int i=0; i < patterns.length; i++) {
|
|
|
|
pp = new ParsePosition(0);
|
|
|
|
Object[] parsed = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
MessageFormat mf = new MessageFormat(patterns[i]);
|
|
|
|
Format[] formats = mf.getFormats();
|
|
|
|
for (int j=0; j < formats.length; j++) {
|
|
|
|
((DecimalFormat)formats[j]).setParseBigDecimal(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
parsed = mf.parse(from[i], pp);
|
|
|
|
|
|
|
|
if (pp.getErrorIndex() != -1) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Case" + (i+1) +
|
2016-05-27 14:33:48 +09:00
|
|
|
": getErrorIndex() returns wrong value. expected:-1, got:"+
|
|
|
|
pp.getErrorIndex() + " for " + from[i]);
|
|
|
|
}
|
|
|
|
if (pp.getIndex() != parsePosition1[i]) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Case" + (i+1) +
|
2016-05-27 14:33:48 +09:00
|
|
|
": getIndex() returns wrong value. expected:" +
|
|
|
|
parsePosition1[i] + ", got:"+ pp.getIndex() +
|
|
|
|
" for " + from[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(Exception e) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Unexpected exception: " + e.getMessage());
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
checkType(from[i], getType(new BigDecimal(expected1[i])),
|
|
|
|
getType((Number)parsed[0]));
|
|
|
|
checkParse(from[i], new BigDecimal(expected1[i]),
|
|
|
|
(Number)parsed[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static final String[] expected2 = { // isParseIntegerOnly() == true
|
|
|
|
"12345678901234567890",
|
|
|
|
"-12345678901234567890",
|
|
|
|
"12345678901234567890",
|
|
|
|
"-12345678901234567890",
|
|
|
|
"12345678901234567890",
|
|
|
|
"-12345678901234567890",
|
|
|
|
"0",
|
|
|
|
"0",
|
|
|
|
|
|
|
|
"9876543210987654321098765432109876543210",
|
|
|
|
"-9876543210987654321098765432109876543210",
|
|
|
|
"9.876543210987654321098765432109876543210E44",
|
|
|
|
"-98765432109876543210987654321098765.43210",
|
|
|
|
"9876543210987654321098765432109876543210",
|
|
|
|
"-9876543210987654321098765432109876543210",
|
|
|
|
"9876543210987654321098765432109876543210.12",
|
|
|
|
"-9876543210987654321098765432109876543210.12",
|
|
|
|
"9876543210987654321098765432109876543210",
|
|
|
|
"-9876543210987654321098765432109876543210.00000000000000000000",
|
|
|
|
};
|
|
|
|
static final int[][] parsePosition2 = { // {errorIndex, index}
|
|
|
|
/*
|
|
|
|
* Should keep in mind that the expected result is different from
|
2024-06-26 17:10:09 +00:00
|
|
|
* DecimalFormat.parse() for some cases. This is because parsing integer
|
|
|
|
* only will return a successful parse for the subformat, but since the index
|
|
|
|
* returned is not equal to the length, at the MessageFormat level, this
|
|
|
|
* will be interpreted as a failed parse, and so the DecimalFormat index
|
|
|
|
* should be reflected as the MessageFormat errorIndex.
|
2016-05-27 14:33:48 +09:00
|
|
|
*/
|
|
|
|
{28, 0}, // parsing stopped at '.'
|
|
|
|
{29, 0}, // parsing stopped at '.'
|
|
|
|
{29, 0}, // parsing stopped at '.'
|
2024-06-26 17:10:09 +00:00
|
|
|
{30, 0}, // parsing stopped at '.'
|
|
|
|
{31, 0}, // parsing stopped at '.'
|
|
|
|
{32, 0}, // parsing stopped at '.'
|
2016-05-27 14:33:48 +09:00
|
|
|
{28, 0}, // parsing stopped at '.'
|
|
|
|
{29, 0}, // parsing stopped at '.'
|
|
|
|
{-1, 57}, {-1, 58}, {-1, 59}, {-1, 61},
|
|
|
|
{56, 0}, // parsing stopped at '.'
|
2024-06-26 17:10:09 +00:00
|
|
|
{57, 0}, // parsing stopped at '.'
|
2016-05-27 14:33:48 +09:00
|
|
|
{-1, 60}, {-1, 61},
|
|
|
|
{28, 0}, // parsing stopped at '.'
|
|
|
|
{-1, 88},
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for MessageFormat: setParseIntegerOnly(true)
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_MessageFormat_ParseIntegerOnly() {
|
2016-05-27 14:33:48 +09:00
|
|
|
for (int i=0; i < patterns.length; i++) {
|
|
|
|
pp = new ParsePosition(0);
|
|
|
|
Object[] parsed = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
MessageFormat mf = new MessageFormat(patterns[i]);
|
|
|
|
Format[] formats = mf.getFormats();
|
|
|
|
for (int j=0; j < formats.length; j++) {
|
|
|
|
((DecimalFormat)formats[j]).setParseBigDecimal(true);
|
|
|
|
((DecimalFormat)formats[j]).setParseIntegerOnly(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
parsed = mf.parse(from[i], pp);
|
|
|
|
|
|
|
|
if (pp.getErrorIndex() != parsePosition2[i][0]) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Case" + (i+1) +
|
2016-05-27 14:33:48 +09:00
|
|
|
": getErrorIndex() returns wrong value. expected:" +
|
|
|
|
parsePosition2[i][0] + ", got:"+ pp.getErrorIndex() +
|
|
|
|
" for " + from[i]);
|
|
|
|
}
|
|
|
|
if (pp.getIndex() != parsePosition2[i][1]) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Case" + (i+1) +
|
2016-05-27 14:33:48 +09:00
|
|
|
": getIndex() returns wrong value. expected:" +
|
|
|
|
parsePosition2[i][1] + ", got:"+ pp.getIndex() +
|
|
|
|
" for " + from[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(Exception e) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Unexpected exception: " + e.getMessage());
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
if (parsePosition2[i][0] == -1) {
|
|
|
|
checkType(from[i], getType(new BigDecimal(expected2[i])),
|
|
|
|
getType((Number)parsed[0]));
|
|
|
|
checkParse(from[i], new BigDecimal(expected2[i]),
|
|
|
|
(Number)parsed[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static final String[] from3 = {
|
|
|
|
"12,345,678,901,234,567,890.98765432109876543210987654321",
|
|
|
|
"-12,345,678,901,234,567,890.98765432109876543210987654321",
|
|
|
|
"9,876,543,210,987,654,321,098,765,432,109,876,543,210",
|
|
|
|
"-9,876,543,210,987,654,321,098,765,432,109,876,543,210",
|
|
|
|
"1234556790000E-8",
|
|
|
|
};
|
|
|
|
static final String[] expected3 = {
|
|
|
|
"12345678901234567890",
|
|
|
|
"-12345678901234567890",
|
|
|
|
"9876543210987654321098765432109876543210",
|
|
|
|
"-9876543210987654321098765432109876543210",
|
|
|
|
"12345.56790000",
|
|
|
|
};
|
|
|
|
static final int[][] parsePosition3 = { // {errorIndex, index}
|
|
|
|
{-1, 26},
|
|
|
|
{-1, 27},
|
|
|
|
{-1, 53},
|
|
|
|
{-1, 54},
|
|
|
|
{-1, 16},
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for DecimalFormat: setParseIntegerOnly(true)
|
|
|
|
*/
|
2023-10-03 16:38:17 +00:00
|
|
|
@Test
|
|
|
|
public void test_Parse_in_DecimalFormat_ParseIntegerOnly() {
|
2016-05-27 14:33:48 +09:00
|
|
|
DecimalFormat df = (DecimalFormat)NumberFormat.getIntegerInstance();
|
|
|
|
df.setParseBigDecimal(true);
|
|
|
|
|
|
|
|
for (int i=0; i < from3.length; i++) {
|
|
|
|
pp = new ParsePosition(0);
|
|
|
|
Number parsed = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
parsed = df.parse(from3[i], pp);
|
|
|
|
|
|
|
|
if (pp.getErrorIndex() != parsePosition3[i][0]) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Case" + (i+1) +
|
2016-05-27 14:33:48 +09:00
|
|
|
": getErrorIndex() returns wrong value. expected:" +
|
|
|
|
parsePosition3[i][0] + ", got:"+ pp.getErrorIndex() +
|
|
|
|
" for " + from3[i]);
|
|
|
|
}
|
|
|
|
if (pp.getIndex() != parsePosition3[i][1]) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Case" + (i+1) +
|
2016-05-27 14:33:48 +09:00
|
|
|
": getIndex() returns wrong value. expected:" +
|
|
|
|
parsePosition3[i][1] + ", got:"+ pp.getIndex() +
|
|
|
|
" for " + from3[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(Exception e) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Unexpected exception: " + e.getMessage());
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
if (parsePosition3[i][0] == -1) {
|
|
|
|
checkType(from3[i], getType(new BigDecimal(expected3[i])),
|
|
|
|
getType(parsed));
|
|
|
|
checkParse(from3[i], new BigDecimal(expected3[i]), parsed);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void check(String from, Number to) {
|
|
|
|
pp = new ParsePosition(0);
|
|
|
|
try {
|
|
|
|
parsed = df.parse(from, pp);
|
|
|
|
}
|
|
|
|
catch(Exception e) {
|
|
|
|
exceptionOccurred = true;
|
2023-10-03 16:38:17 +00:00
|
|
|
fail(e.getMessage());
|
2016-05-27 14:33:48 +09:00
|
|
|
}
|
|
|
|
if (!exceptionOccurred) {
|
|
|
|
checkParse(from, to, parsed);
|
|
|
|
checkType(from, getType(to), getType(parsed));
|
|
|
|
checkParsePosition(from, from.length(), pp.getIndex());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void checkParse(String orig, Number expected, Number got) {
|
|
|
|
if (!expected.equals(got)) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Parsing... failed." +
|
2016-05-27 14:33:48 +09:00
|
|
|
"\n original: " + orig +
|
|
|
|
"\n parsed: " + got +
|
|
|
|
"\n expected: " + expected + "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void checkType(String orig, String expected, String got) {
|
|
|
|
if (!expected.equals(got)) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Parsing... unexpected Class returned." +
|
2016-05-27 14:33:48 +09:00
|
|
|
"\n original: " + orig +
|
|
|
|
"\n got: " + got +
|
|
|
|
"\n expected: " + expected + "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void checkParsePosition(String orig, int expected, int got) {
|
|
|
|
if (expected != got) {
|
2023-10-03 16:38:17 +00:00
|
|
|
fail("Parsing... wrong ParsePosition returned." +
|
2016-05-27 14:33:48 +09:00
|
|
|
"\n original: " + orig +
|
|
|
|
"\n got: " + got +
|
|
|
|
"\n expected: " + expected + "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private String getType(Number number) {
|
|
|
|
return number.getClass().getName();
|
|
|
|
}
|
|
|
|
}
|