ICU-12874 Don't truncate small numbers with CompactDecimalFormat style rules represented in RBNF

X-SVN-Rev: 39557
This commit is contained in:
George Rhoten 2017-01-13 19:06:54 +00:00
parent 40f9b33eb7
commit 0b24b6a633
2 changed files with 72 additions and 11 deletions

View File

@ -278,6 +278,8 @@ abstract class NFSubstitution {
// formatting
//-----------------------------------------------------------------------
private static final long MAX_INT64_IN_DOUBLE = 0x1FFFFFFFFFFFFFL;
/**
* Performs a mathematical operation on the number, formats it using
* either ruleSet or decimalFormat, and inserts the result into
@ -289,16 +291,38 @@ abstract class NFSubstitution {
* position to determine exactly where to insert the new text)
*/
public void doSubstitution(long number, StringBuilder toInsertInto, int position, int recursionCount) {
// perform a transformation on the number that is dependent
if (ruleSet != null) {
// Perform a transformation on the number that is dependent
// on the type of substitution this is, then just call its
// rule set's format() method to format the result
long numberToFormat = transformNumber(number);
if (ruleSet != null) {
ruleSet.format(numberToFormat, toInsertInto, position + pos, recursionCount);
} else {
if (number <= MAX_INT64_IN_DOUBLE) {
// or perform the transformation on the number (preserving
// the result's fractional part if the formatter it set
// to show it), then use that formatter's format() method
// to format the result
double numberToFormat = transformNumber((double) number);
if (numberFormat.getMaximumFractionDigits() == 0) {
numberToFormat = Math.floor(numberToFormat);
}
toInsertInto.insert(position + pos, numberFormat.format(numberToFormat));
}
else {
// We have gone beyond double precision. Something has to give.
// We're favoring accuracy of the large number over potential rules
// that round like a CompactDecimalFormat, which is not a common use case.
//
// Perform a transformation on the number that is dependent
// on the type of substitution this is, then just call its
// rule set's format() method to format the result
long numberToFormat = transformNumber(number);
toInsertInto.insert(position + pos, numberFormat.format(numberToFormat));
}
}
}
/**

View File

@ -948,9 +948,9 @@ public class RbnfTest extends TestFmwk {
@Test
public void TestRuleSetDisplayName() {
/**
/*
* Spellout rules for U.K. English.
* I borrow the rule sets for TestRuleSetDisplayName()
* This was borrowed from the rule sets for TestRuleSetDisplayName()
*/
final String ukEnglish =
"%simplified:\n"
@ -1309,13 +1309,13 @@ public class RbnfTest extends TestFmwk {
rbnf.getRuleSetDisplayName("", new ULocale("en_US"));
errln("RuleBasedNumberFormat.getRuleSetDisplayName(String ruleSetName, ULocale loc) " +
"was suppose to have an exception.");
} catch(Exception e){}
} catch(Exception ignored){}
try{
rbnf.getRuleSetDisplayName("dummy", new ULocale("en_US"));
errln("RuleBasedNumberFormat.getRuleSetDisplayName(String ruleSetName, ULocale loc) " +
"was suppose to have an exception.");
} catch(Exception e){}
} catch(Exception ignored){}
}
/* Test the method
@ -1651,6 +1651,10 @@ public class RbnfTest extends TestFmwk {
RuleBasedNumberFormat rbnf = new RuleBasedNumberFormat(ULocale.US, RuleBasedNumberFormat.SPELLOUT);
String[][] enTestFullData = {
{"-9007199254740991", "minus nine quadrillion seven trillion one hundred ninety-nine billion two hundred fifty-four million seven hundred forty thousand nine hundred ninety-one"}, // Maximum precision in both a double and a long
{"9007199254740991", "nine quadrillion seven trillion one hundred ninety-nine billion two hundred fifty-four million seven hundred forty thousand nine hundred ninety-one"}, // Maximum precision in both a double and a long
{"-9007199254740992", "minus nine quadrillion seven trillion one hundred ninety-nine billion two hundred fifty-four million seven hundred forty thousand nine hundred ninety-two"}, // Only precisely contained in a long
{"9007199254740992", "nine quadrillion seven trillion one hundred ninety-nine billion two hundred fifty-four million seven hundred forty thousand nine hundred ninety-two"}, // Only precisely contained in a long
{"9999999999999998", "nine quadrillion nine hundred ninety-nine trillion nine hundred ninety-nine billion nine hundred ninety-nine million nine hundred ninety-nine thousand nine hundred ninety-eight"},
{"9999999999999999", "nine quadrillion nine hundred ninety-nine trillion nine hundred ninety-nine billion nine hundred ninety-nine million nine hundred ninety-nine thousand nine hundred ninety-nine"},
{"999999999999999999", "nine hundred ninety-nine quadrillion nine hundred ninety-nine trillion nine hundred ninety-nine billion nine hundred ninety-nine million nine hundred ninety-nine thousand nine hundred ninety-nine"},
@ -1664,7 +1668,40 @@ public class RbnfTest extends TestFmwk {
{"9223372036854775000", "9,223,372,036,854,775,000"}, // Below 64-bit precision
{"9223372036854775806", "9,223,372,036,854,775,806"}, // Maximum 64-bit precision - 1
{"9223372036854775807", "9,223,372,036,854,775,807"}, // Maximum 64-bit precision
{"9223372036854775808", "9,223,372,036,854,775,808"}, // We've gone beyond 64-bit precision
{"9223372036854775808", "9,223,372,036,854,775,808"}, // We've gone beyond 64-bit precision. This can only be represented with BigDecimal.
};
doTest(rbnf, enTestFullData, false);
}
@Test
public void testCompactDecimalFormatStyle() {
// This is not a common use case, but we're testing it anyway.
final String numberPattern = "=###0.#####=;"
+ "1000: <###0.00< K;"
+ "1000000: <###0.00< M;"
+ "1000000000: <###0.00< B;"
+ "1000000000000: <###0.00< T;"
+ "1000000000000000: <###0.00< Q;";
RuleBasedNumberFormat rbnf = new RuleBasedNumberFormat(numberPattern, ULocale.US);
String[][] enTestFullData = {
{"1000", "1.00 K"},
{"1234", "1.23 K"},
{"999994", "999.99 K"},
{"999995", "1000.00 K"},
{"1000000", "1.00 M"},
{"1200000", "1.20 M"},
{"1200000000", "1.20 B"},
{"1200000000000", "1.20 T"},
{"1200000000000000", "1.20 Q"},
{"4503599627370495", "4.50 Q"},
{"4503599627370496", "4.50 Q"},
{"8990000000000000", "8.99 Q"},
{"9008000000000000", "9.00 Q"}, // Number doesn't precisely fit into a double
{"9456000000000000", "9.00 Q"}, // Number doesn't precisely fit into a double
{"10000000000000000", "10.00 Q"}, // Number doesn't precisely fit into a double
{"9223372036854775807", "9223.00 Q"}, // Maximum 64-bit precision
{"9223372036854775808", "9,223,372,036,854,775,808"}, // We've gone beyond 64-bit precision. This can only be represented with BigDecimal.
};
doTest(rbnf, enTestFullData, false);
}