1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * Copyright (c) 1996, 2010, Oracle and/or its affiliates. All rights reserved. 4 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 5 * 6 * This code is free software; you can redistribute it and/or modify it 7 * under the terms of the GNU General Public License version 2 only, as 8 * published by the Free Software Foundation. Oracle designates this 9 * particular file as subject to the "Classpath" exception as provided 10 * by Oracle in the LICENSE file that accompanied this code. 11 * 12 * This code is distributed in the hope that it will be useful, but WITHOUT 13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 15 * version 2 for more details (a copy is included in the LICENSE file that 16 * accompanied this code). 17 * 18 * You should have received a copy of the GNU General Public License version 19 * 2 along with this work; if not, write to the Free Software Foundation, 20 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 21 * 22 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 23 * or visit www.oracle.com if you need additional information or have any 24 * questions. 25 */ 26 27 /* 28 * (C) Copyright Taligent, Inc. 1996, 1997 - All Rights Reserved 29 * (C) Copyright IBM Corp. 1996 - 1998 - All Rights Reserved 30 * 31 * The original version of this source code and documentation is copyrighted 32 * and owned by Taligent, Inc., a wholly-owned subsidiary of IBM. These 33 * materials are provided under terms of a License Agreement between Taligent 34 * and Sun. This technology is protected by multiple US and International 35 * patents. This notice and attribution to Taligent may not be removed. 36 * Taligent is a registered trademark of Taligent, Inc. 37 * 38 */ 39 40 package java.text; 41 42 import java.io.IOException; 43 import java.io.ObjectInputStream; 44 import java.io.ObjectOutputStream; 45 import java.io.ObjectStreamField; 46 import java.math.BigDecimal; 47 import java.math.BigInteger; 48 import java.math.RoundingMode; 49 import java.util.Currency; 50 import java.util.Locale; 51 import java.util.concurrent.ConcurrentHashMap; 52 import java.util.concurrent.ConcurrentMap; 53 import java.util.concurrent.atomic.AtomicInteger; 54 import java.util.concurrent.atomic.AtomicLong; 55 import libcore.icu.LocaleData; 56 57 import android.icu.math.MathContext; 58 59 /** 60 * <code>DecimalFormat</code> is a concrete subclass of 61 * <code>NumberFormat</code> that formats decimal numbers. It has a variety of 62 * features designed to make it possible to parse and format numbers in any 63 * locale, including support for Western, Arabic, and Indic digits. It also 64 * supports different kinds of numbers, including integers (123), fixed-point 65 * numbers (123.4), scientific notation (1.23E4), percentages (12%), and 66 * currency amounts ($123). All of these can be localized. 67 * 68 * <p>To obtain a <code>NumberFormat</code> for a specific locale, including the 69 * default locale, call one of <code>NumberFormat</code>'s factory methods, such 70 * as <code>getInstance()</code>. In general, do not call the 71 * <code>DecimalFormat</code> constructors directly, since the 72 * <code>NumberFormat</code> factory methods may return subclasses other than 73 * <code>DecimalFormat</code>. If you need to customize the format object, do 74 * something like this: 75 * 76 * <blockquote><pre> 77 * NumberFormat f = NumberFormat.getInstance(loc); 78 * if (f instanceof DecimalFormat) { 79 * ((DecimalFormat) f).setDecimalSeparatorAlwaysShown(true); 80 * } 81 * </pre></blockquote> 82 * 83 * <p>A <code>DecimalFormat</code> comprises a <em>pattern</em> and a set of 84 * <em>symbols</em>. The pattern may be set directly using 85 * <code>applyPattern()</code>, or indirectly using the API methods. The 86 * symbols are stored in a <code>DecimalFormatSymbols</code> object. When using 87 * the <code>NumberFormat</code> factory methods, the pattern and symbols are 88 * read from localized <code>ResourceBundle</code>s. 89 * 90 * <h4>Patterns</h4> 91 * 92 * <code>DecimalFormat</code> patterns have the following syntax: 93 * <blockquote><pre> 94 * <i>Pattern:</i> 95 * <i>PositivePattern</i> 96 * <i>PositivePattern</i> ; <i>NegativePattern</i> 97 * <i>PositivePattern:</i> 98 * <i>Prefix<sub>opt</sub></i> <i>Number</i> <i>Suffix<sub>opt</sub></i> 99 * <i>NegativePattern:</i> 100 * <i>Prefix<sub>opt</sub></i> <i>Number</i> <i>Suffix<sub>opt</sub></i> 101 * <i>Prefix:</i> 102 * any Unicode characters except \uFFFE, \uFFFF, and special characters 103 * <i>Suffix:</i> 104 * any Unicode characters except \uFFFE, \uFFFF, and special characters 105 * <i>Number:</i> 106 * <i>Integer</i> <i>Exponent<sub>opt</sub></i> 107 * <i>Integer</i> . <i>Fraction</i> <i>Exponent<sub>opt</sub></i> 108 * <i>Integer:</i> 109 * <i>MinimumInteger</i> 110 * # 111 * # <i>Integer</i> 112 * # , <i>Integer</i> 113 * <i>MinimumInteger:</i> 114 * 0 115 * 0 <i>MinimumInteger</i> 116 * 0 , <i>MinimumInteger</i> 117 * <i>Fraction:</i> 118 * <i>MinimumFraction<sub>opt</sub></i> <i>OptionalFraction<sub>opt</sub></i> 119 * <i>MinimumFraction:</i> 120 * 0 <i>MinimumFraction<sub>opt</sub></i> 121 * <i>OptionalFraction:</i> 122 * # <i>OptionalFraction<sub>opt</sub></i> 123 * <i>Exponent:</i> 124 * E <i>MinimumExponent</i> 125 * <i>MinimumExponent:</i> 126 * 0 <i>MinimumExponent<sub>opt</sub></i> 127 * </pre></blockquote> 128 * 129 * <p>A <code>DecimalFormat</code> pattern contains a positive and negative 130 * subpattern, for example, <code>"#,##0.00;(#,##0.00)"</code>. Each 131 * subpattern has a prefix, numeric part, and suffix. The negative subpattern 132 * is optional; if absent, then the positive subpattern prefixed with the 133 * localized minus sign (<code>'-'</code> in most locales) is used as the 134 * negative subpattern. That is, <code>"0.00"</code> alone is equivalent to 135 * <code>"0.00;-0.00"</code>. If there is an explicit negative subpattern, it 136 * serves only to specify the negative prefix and suffix; the number of digits, 137 * minimal digits, and other characteristics are all the same as the positive 138 * pattern. That means that <code>"#,##0.0#;(#)"</code> produces precisely 139 * the same behavior as <code>"#,##0.0#;(#,##0.0#)"</code>. 140 * 141 * <p>The prefixes, suffixes, and various symbols used for infinity, digits, 142 * thousands separators, decimal separators, etc. may be set to arbitrary 143 * values, and they will appear properly during formatting. However, care must 144 * be taken that the symbols and strings do not conflict, or parsing will be 145 * unreliable. For example, either the positive and negative prefixes or the 146 * suffixes must be distinct for <code>DecimalFormat.parse()</code> to be able 147 * to distinguish positive from negative values. (If they are identical, then 148 * <code>DecimalFormat</code> will behave as if no negative subpattern was 149 * specified.) Another example is that the decimal separator and thousands 150 * separator should be distinct characters, or parsing will be impossible. 151 * 152 * <p>The grouping separator is commonly used for thousands, but in some 153 * countries it separates ten-thousands. The grouping size is a constant number 154 * of digits between the grouping characters, such as 3 for 100,000,000 or 4 for 155 * 1,0000,0000. If you supply a pattern with multiple grouping characters, the 156 * interval between the last one and the end of the integer is the one that is 157 * used. So <code>"#,##,###,####"</code> == <code>"######,####"</code> == 158 * <code>"##,####,####"</code>. 159 * 160 * <h4>Special Pattern Characters</h4> 161 * 162 * <p>Many characters in a pattern are taken literally; they are matched during 163 * parsing and output unchanged during formatting. Special characters, on the 164 * other hand, stand for other characters, strings, or classes of characters. 165 * They must be quoted, unless noted otherwise, if they are to appear in the 166 * prefix or suffix as literals. 167 * 168 * <p>The characters listed here are used in non-localized patterns. Localized 169 * patterns use the corresponding characters taken from this formatter's 170 * <code>DecimalFormatSymbols</code> object instead, and these characters lose 171 * their special status. Two exceptions are the currency sign and quote, which 172 * are not localized. 173 * 174 * <blockquote> 175 * <table border=0 cellspacing=3 cellpadding=0 summary="Chart showing symbol, 176 * location, localized, and meaning."> 177 * <tr bgcolor="#ccccff"> 178 * <th align=left>Symbol 179 * <th align=left>Location 180 * <th align=left>Localized? 181 * <th align=left>Meaning 182 * <tr valign=top> 183 * <td><code>0</code> 184 * <td>Number 185 * <td>Yes 186 * <td>Digit 187 * <tr valign=top bgcolor="#eeeeff"> 188 * <td><code>#</code> 189 * <td>Number 190 * <td>Yes 191 * <td>Digit, zero shows as absent 192 * <tr valign=top> 193 * <td><code>.</code> 194 * <td>Number 195 * <td>Yes 196 * <td>Decimal separator or monetary decimal separator 197 * <tr valign=top bgcolor="#eeeeff"> 198 * <td><code>-</code> 199 * <td>Number 200 * <td>Yes 201 * <td>Minus sign 202 * <tr valign=top> 203 * <td><code>,</code> 204 * <td>Number 205 * <td>Yes 206 * <td>Grouping separator 207 * <tr valign=top bgcolor="#eeeeff"> 208 * <td><code>E</code> 209 * <td>Number 210 * <td>Yes 211 * <td>Separates mantissa and exponent in scientific notation. 212 * <em>Need not be quoted in prefix or suffix.</em> 213 * <tr valign=top> 214 * <td><code>;</code> 215 * <td>Subpattern boundary 216 * <td>Yes 217 * <td>Separates positive and negative subpatterns 218 * <tr valign=top bgcolor="#eeeeff"> 219 * <td><code>%</code> 220 * <td>Prefix or suffix 221 * <td>Yes 222 * <td>Multiply by 100 and show as percentage 223 * <tr valign=top> 224 * <td><code>\u2030</code> 225 * <td>Prefix or suffix 226 * <td>Yes 227 * <td>Multiply by 1000 and show as per mille value 228 * <tr valign=top bgcolor="#eeeeff"> 229 * <td><code>¤</code> (<code>\u00A4</code>) 230 * <td>Prefix or suffix 231 * <td>No 232 * <td>Currency sign, replaced by currency symbol. If 233 * doubled, replaced by international currency symbol. 234 * If present in a pattern, the monetary decimal separator 235 * is used instead of the decimal separator. 236 * <tr valign=top> 237 * <td><code>'</code> 238 * <td>Prefix or suffix 239 * <td>No 240 * <td>Used to quote special characters in a prefix or suffix, 241 * for example, <code>"'#'#"</code> formats 123 to 242 * <code>"#123"</code>. To create a single quote 243 * itself, use two in a row: <code>"# o''clock"</code>. 244 * </table> 245 * </blockquote> 246 * 247 * <h4>Scientific Notation</h4> 248 * 249 * <p>Numbers in scientific notation are expressed as the product of a mantissa 250 * and a power of ten, for example, 1234 can be expressed as 1.234 x 10^3. The 251 * mantissa is often in the range 1.0 <= x < 10.0, but it need not be. 252 * <code>DecimalFormat</code> can be instructed to format and parse scientific 253 * notation <em>only via a pattern</em>; there is currently no factory method 254 * that creates a scientific notation format. In a pattern, the exponent 255 * character immediately followed by one or more digit characters indicates 256 * scientific notation. Example: <code>"0.###E0"</code> formats the number 257 * 1234 as <code>"1.234E3"</code>. 258 * 259 * <ul> 260 * <li>The number of digit characters after the exponent character gives the 261 * minimum exponent digit count. There is no maximum. Negative exponents are 262 * formatted using the localized minus sign, <em>not</em> the prefix and suffix 263 * from the pattern. This allows patterns such as <code>"0.###E0 m/s"</code>. 264 * 265 * <li>The minimum and maximum number of integer digits are interpreted 266 * together: 267 * 268 * <ul> 269 * <li>If the maximum number of integer digits is greater than their minimum number 270 * and greater than 1, it forces the exponent to be a multiple of the maximum 271 * number of integer digits, and the minimum number of integer digits to be 272 * interpreted as 1. The most common use of this is to generate 273 * <em>engineering notation</em>, in which the exponent is a multiple of three, 274 * e.g., <code>"##0.#####E0"</code>. Using this pattern, the number 12345 275 * formats to <code>"12.345E3"</code>, and 123456 formats to 276 * <code>"123.456E3"</code>. 277 * 278 * <li>Otherwise, the minimum number of integer digits is achieved by adjusting the 279 * exponent. Example: 0.00123 formatted with <code>"00.###E0"</code> yields 280 * <code>"12.3E-4"</code>. 281 * </ul> 282 * 283 * <li>The number of significant digits in the mantissa is the sum of the 284 * <em>minimum integer</em> and <em>maximum fraction</em> digits, and is 285 * unaffected by the maximum integer digits. For example, 12345 formatted with 286 * <code>"##0.##E0"</code> is <code>"12.3E3"</code>. To show all digits, set 287 * the significant digits count to zero. The number of significant digits 288 * does not affect parsing. 289 * 290 * <li>Exponential patterns may not contain grouping separators. 291 * </ul> 292 * 293 * <h4>Rounding</h4> 294 * 295 * <code>DecimalFormat</code> provides rounding modes defined in 296 * {@link java.math.RoundingMode} for formatting. By default, it uses 297 * {@link java.math.RoundingMode#HALF_EVEN RoundingMode.HALF_EVEN}. 298 * 299 * <h4>Digits</h4> 300 * 301 * For formatting, <code>DecimalFormat</code> uses the ten consecutive 302 * characters starting with the localized zero digit defined in the 303 * <code>DecimalFormatSymbols</code> object as digits. For parsing, these 304 * digits as well as all Unicode decimal digits, as defined by 305 * {@link Character#digit Character.digit}, are recognized. 306 * 307 * <h4>Special Values</h4> 308 * 309 * <p><code>NaN</code> is formatted as a string, which typically has a single character 310 * <code>\uFFFD</code>. This string is determined by the 311 * <code>DecimalFormatSymbols</code> object. This is the only value for which 312 * the prefixes and suffixes are not used. 313 * 314 * <p>Infinity is formatted as a string, which typically has a single character 315 * <code>\u221E</code>, with the positive or negative prefixes and suffixes 316 * applied. The infinity string is determined by the 317 * <code>DecimalFormatSymbols</code> object. 318 * 319 * <p>Negative zero (<code>"-0"</code>) parses to 320 * <ul> 321 * <li><code>BigDecimal(0)</code> if <code>isParseBigDecimal()</code> is 322 * true, 323 * <li><code>Long(0)</code> if <code>isParseBigDecimal()</code> is false 324 * and <code>isParseIntegerOnly()</code> is true, 325 * <li><code>Double(-0.0)</code> if both <code>isParseBigDecimal()</code> 326 * and <code>isParseIntegerOnly()</code> are false. 327 * </ul> 328 * 329 * <h4><a name="synchronization">Synchronization</a></h4> 330 * 331 * <p> 332 * Decimal formats are generally not synchronized. 333 * It is recommended to create separate format instances for each thread. 334 * If multiple threads access a format concurrently, it must be synchronized 335 * externally. 336 * 337 * <h4>Example</h4> 338 * 339 * <blockquote><pre> 340 * <strong>// Print out a number using the localized number, integer, currency, 341 * // and percent format for each locale</strong> 342 * Locale[] locales = NumberFormat.getAvailableLocales(); 343 * double myNumber = -1234.56; 344 * NumberFormat form; 345 * for (int j=0; j<4; ++j) { 346 * System.out.println("FORMAT"); 347 * for (int i = 0; i < locales.length; ++i) { 348 * if (locales[i].getCountry().length() == 0) { 349 * continue; // Skip language-only locales 350 * } 351 * System.out.print(locales[i].getDisplayName()); 352 * switch (j) { 353 * case 0: 354 * form = NumberFormat.getInstance(locales[i]); break; 355 * case 1: 356 * form = NumberFormat.getIntegerInstance(locales[i]); break; 357 * case 2: 358 * form = NumberFormat.getCurrencyInstance(locales[i]); break; 359 * default: 360 * form = NumberFormat.getPercentInstance(locales[i]); break; 361 * } 362 * if (form instanceof DecimalFormat) { 363 * System.out.print(": " + ((DecimalFormat) form).toPattern()); 364 * } 365 * System.out.print(" -> " + form.format(myNumber)); 366 * try { 367 * System.out.println(" -> " + form.parse(form.format(myNumber))); 368 * } catch (ParseException e) {} 369 * } 370 * } 371 * </pre></blockquote> 372 * 373 * @see <a href="http://java.sun.com/docs/books/tutorial/i18n/format/decimalFormat.html">Java Tutorial</a> 374 * @see NumberFormat 375 * @see DecimalFormatSymbols 376 * @see ParsePosition 377 * @author Mark Davis 378 * @author Alan Liu 379 */ 380 public class DecimalFormat extends NumberFormat { 381 382 private transient android.icu.text.DecimalFormat icuDecimalFormat; 383 384 /** 385 * Creates a DecimalFormat using the default pattern and symbols 386 * for the default locale. This is a convenient way to obtain a 387 * DecimalFormat when internationalization is not the main concern. 388 * <p> 389 * To obtain standard formats for a given locale, use the factory methods 390 * on NumberFormat such as getNumberInstance. These factories will 391 * return the most appropriate sub-class of NumberFormat for a given 392 * locale. 393 * 394 * @see java.text.NumberFormat#getInstance 395 * @see java.text.NumberFormat#getNumberInstance 396 * @see java.text.NumberFormat#getCurrencyInstance 397 * @see java.text.NumberFormat#getPercentInstance 398 */ DecimalFormat()399 public DecimalFormat() { 400 Locale def = Locale.getDefault(Locale.Category.FORMAT); 401 // try to get the pattern from the cache 402 String pattern = cachedLocaleData.get(def); 403 if (pattern == null) { /* cache miss */ 404 // Get the pattern for the default locale. 405 pattern = LocaleData.get(def).numberPattern; 406 /* update cache */ 407 cachedLocaleData.putIfAbsent(def, pattern); 408 } 409 this.symbols = new DecimalFormatSymbols(def); 410 init(pattern); 411 } 412 413 414 /** 415 * Creates a DecimalFormat using the given pattern and the symbols 416 * for the default locale. This is a convenient way to obtain a 417 * DecimalFormat when internationalization is not the main concern. 418 * <p> 419 * To obtain standard formats for a given locale, use the factory methods 420 * on NumberFormat such as getNumberInstance. These factories will 421 * return the most appropriate sub-class of NumberFormat for a given 422 * locale. 423 * 424 * @param pattern A non-localized pattern string. 425 * @exception NullPointerException if <code>pattern</code> is null 426 * @exception IllegalArgumentException if the given pattern is invalid. 427 * @see java.text.NumberFormat#getInstance 428 * @see java.text.NumberFormat#getNumberInstance 429 * @see java.text.NumberFormat#getCurrencyInstance 430 * @see java.text.NumberFormat#getPercentInstance 431 */ DecimalFormat(String pattern)432 public DecimalFormat(String pattern) { 433 this.symbols = new DecimalFormatSymbols(Locale.getDefault(Locale.Category.FORMAT)); 434 init(pattern); 435 } 436 437 438 /** 439 * Creates a DecimalFormat using the given pattern and symbols. 440 * Use this constructor when you need to completely customize the 441 * behavior of the format. 442 * <p> 443 * To obtain standard formats for a given 444 * locale, use the factory methods on NumberFormat such as 445 * getInstance or getCurrencyInstance. If you need only minor adjustments 446 * to a standard format, you can modify the format returned by 447 * a NumberFormat factory method. 448 * 449 * @param pattern a non-localized pattern string 450 * @param symbols the set of symbols to be used 451 * @exception NullPointerException if any of the given arguments is null 452 * @exception IllegalArgumentException if the given pattern is invalid 453 * @see java.text.NumberFormat#getInstance 454 * @see java.text.NumberFormat#getNumberInstance 455 * @see java.text.NumberFormat#getCurrencyInstance 456 * @see java.text.NumberFormat#getPercentInstance 457 * @see java.text.DecimalFormatSymbols 458 */ DecimalFormat(String pattern, DecimalFormatSymbols symbols)459 public DecimalFormat (String pattern, DecimalFormatSymbols symbols) { 460 // Always applyPattern after the symbols are set 461 this.symbols = (DecimalFormatSymbols)symbols.clone(); 462 init(pattern); 463 } 464 init(String pattern)465 private void init(String pattern) { 466 this.icuDecimalFormat = new android.icu.text.DecimalFormat(pattern, 467 symbols.getIcuDecimalFormatSymbols()); 468 updateFieldsFromIcu(); 469 } 470 471 /** 472 * Converts between field positions used by Java/ICU. 473 * @param fp The java.text.NumberFormat.Field field position 474 * @return The android.icu.text.NumberFormat.Field field position 475 */ getIcuFieldPosition(FieldPosition fp)476 private static FieldPosition getIcuFieldPosition(FieldPosition fp) { 477 if (fp.getFieldAttribute() == null) return fp; 478 479 android.icu.text.NumberFormat.Field attribute; 480 if (fp.getFieldAttribute() == Field.INTEGER) { 481 attribute = android.icu.text.NumberFormat.Field.INTEGER; 482 } else if (fp.getFieldAttribute() == Field.FRACTION) { 483 attribute = android.icu.text.NumberFormat.Field.FRACTION; 484 } else if (fp.getFieldAttribute() == Field.DECIMAL_SEPARATOR) { 485 attribute = android.icu.text.NumberFormat.Field.DECIMAL_SEPARATOR; 486 } else if (fp.getFieldAttribute() == Field.EXPONENT_SYMBOL) { 487 attribute = android.icu.text.NumberFormat.Field.EXPONENT_SYMBOL; 488 } else if (fp.getFieldAttribute() == Field.EXPONENT_SIGN) { 489 attribute = android.icu.text.NumberFormat.Field.EXPONENT_SIGN; 490 } else if (fp.getFieldAttribute() == Field.EXPONENT) { 491 attribute = android.icu.text.NumberFormat.Field.EXPONENT; 492 } else if (fp.getFieldAttribute() == Field.GROUPING_SEPARATOR) { 493 attribute = android.icu.text.NumberFormat.Field.GROUPING_SEPARATOR; 494 } else if (fp.getFieldAttribute() == Field.CURRENCY) { 495 attribute = android.icu.text.NumberFormat.Field.CURRENCY; 496 } else if (fp.getFieldAttribute() == Field.PERCENT) { 497 attribute = android.icu.text.NumberFormat.Field.PERCENT; 498 } else if (fp.getFieldAttribute() == Field.PERMILLE) { 499 attribute = android.icu.text.NumberFormat.Field.PERMILLE; 500 } else if (fp.getFieldAttribute() == Field.SIGN) { 501 attribute = android.icu.text.NumberFormat.Field.SIGN; 502 } else { 503 throw new IllegalArgumentException("Unexpected field position attribute type."); 504 } 505 506 FieldPosition icuFieldPosition = new FieldPosition(attribute); 507 icuFieldPosition.setBeginIndex(fp.getBeginIndex()); 508 icuFieldPosition.setEndIndex(fp.getEndIndex()); 509 return icuFieldPosition; 510 } 511 512 /** 513 * Converts the Attribute that ICU returns in its AttributedCharacterIterator 514 * responses to the type that java uses. 515 * @param icuAttribute The AttributedCharacterIterator.Attribute field. 516 * @return Field converted to a java.text.NumberFormat.Field field. 517 */ toJavaFieldAttribute(AttributedCharacterIterator.Attribute icuAttribute)518 private static Field toJavaFieldAttribute(AttributedCharacterIterator.Attribute icuAttribute) { 519 if (icuAttribute.getName().equals(Field.INTEGER.getName())) { 520 return Field.INTEGER; 521 } 522 if (icuAttribute.getName().equals(Field.CURRENCY.getName())) { 523 return Field.CURRENCY; 524 } 525 if (icuAttribute.getName().equals(Field.DECIMAL_SEPARATOR.getName())) { 526 return Field.DECIMAL_SEPARATOR; 527 } 528 if (icuAttribute.getName().equals(Field.EXPONENT.getName())) { 529 return Field.EXPONENT; 530 } 531 if (icuAttribute.getName().equals(Field.EXPONENT_SIGN.getName())) { 532 return Field.EXPONENT_SIGN; 533 } 534 if (icuAttribute.getName().equals(Field.EXPONENT_SYMBOL.getName())) { 535 return Field.EXPONENT_SYMBOL; 536 } 537 if (icuAttribute.getName().equals(Field.FRACTION.getName())) { 538 return Field.FRACTION; 539 } 540 if (icuAttribute.getName().equals(Field.GROUPING_SEPARATOR.getName())) { 541 return Field.GROUPING_SEPARATOR; 542 } 543 if (icuAttribute.getName().equals(Field.SIGN.getName())) { 544 return Field.SIGN; 545 } 546 if (icuAttribute.getName().equals(Field.PERCENT.getName())) { 547 return Field.PERCENT; 548 } 549 if (icuAttribute.getName().equals(Field.PERMILLE.getName())) { 550 return Field.PERMILLE; 551 } 552 throw new IllegalArgumentException("Unrecognized attribute: " + icuAttribute.getName()); 553 } 554 555 // Overrides 556 /** 557 * Formats a number and appends the resulting text to the given string 558 * buffer. 559 * The number can be of any subclass of {@link java.lang.Number}. 560 * <p> 561 * This implementation uses the maximum precision permitted. 562 * @param number the number to format 563 * @param toAppendTo the <code>StringBuffer</code> to which the formatted 564 * text is to be appended 565 * @param pos On input: an alignment field, if desired. 566 * On output: the offsets of the alignment field. 567 * @return the value passed in as <code>toAppendTo</code> 568 * @exception IllegalArgumentException if <code>number</code> is 569 * null or not an instance of <code>Number</code>. 570 * @exception NullPointerException if <code>toAppendTo</code> or 571 * <code>pos</code> is null 572 * @exception ArithmeticException if rounding is needed with rounding 573 * mode being set to RoundingMode.UNNECESSARY 574 * @see java.text.FieldPosition 575 */ format(Object number, StringBuffer toAppendTo, FieldPosition pos)576 public final StringBuffer format(Object number, 577 StringBuffer toAppendTo, 578 FieldPosition pos) { 579 if (number instanceof Long || number instanceof Integer || 580 number instanceof Short || number instanceof Byte || 581 number instanceof AtomicInteger || 582 number instanceof AtomicLong || 583 (number instanceof BigInteger && 584 ((BigInteger)number).bitLength () < 64)) { 585 return format(((Number)number).longValue(), toAppendTo, pos); 586 } else if (number instanceof BigDecimal) { 587 return format((BigDecimal)number, toAppendTo, pos); 588 } else if (number instanceof BigInteger) { 589 return format((BigInteger)number, toAppendTo, pos); 590 } else if (number instanceof Number) { 591 return format(((Number)number).doubleValue(), toAppendTo, pos); 592 } else { 593 throw new IllegalArgumentException("Cannot format given Object as a Number"); 594 } 595 } 596 597 /** 598 * Formats a double to produce a string. 599 * @param number The double to format 600 * @param result where the text is to be appended 601 * @param fieldPosition On input: an alignment field, if desired. 602 * On output: the offsets of the alignment field. 603 * @exception ArithmeticException if rounding is needed with rounding 604 * mode being set to RoundingMode.UNNECESSARY 605 * @return The formatted number string 606 * @see java.text.FieldPosition 607 */ format(double number, StringBuffer result, FieldPosition fieldPosition)608 public StringBuffer format(double number, StringBuffer result, 609 FieldPosition fieldPosition) { 610 FieldPosition icuFieldPosition = getIcuFieldPosition(fieldPosition); 611 icuDecimalFormat.format(number, result, icuFieldPosition); 612 fieldPosition.setBeginIndex(icuFieldPosition.getBeginIndex()); 613 fieldPosition.setEndIndex(icuFieldPosition.getEndIndex()); 614 return result; 615 } 616 617 /** 618 * Format a long to produce a string. 619 * @param number The long to format 620 * @param result where the text is to be appended 621 * @param fieldPosition On input: an alignment field, if desired. 622 * On output: the offsets of the alignment field. 623 * @exception ArithmeticException if rounding is needed with rounding 624 * mode being set to RoundingMode.UNNECESSARY 625 * @return The formatted number string 626 * @see java.text.FieldPosition 627 */ format(long number, StringBuffer result, FieldPosition fieldPosition)628 public StringBuffer format(long number, StringBuffer result, 629 FieldPosition fieldPosition) { 630 FieldPosition icuFieldPosition = getIcuFieldPosition(fieldPosition); 631 icuDecimalFormat.format(number, result, icuFieldPosition); 632 fieldPosition.setBeginIndex(icuFieldPosition.getBeginIndex()); 633 fieldPosition.setEndIndex(icuFieldPosition.getEndIndex()); 634 return result; 635 } 636 637 /** 638 * Formats a BigDecimal to produce a string. 639 * @param number The BigDecimal to format 640 * @param result where the text is to be appended 641 * @param fieldPosition On input: an alignment field, if desired. 642 * On output: the offsets of the alignment field. 643 * @return The formatted number string 644 * @exception ArithmeticException if rounding is needed with rounding 645 * mode being set to RoundingMode.UNNECESSARY 646 * @see java.text.FieldPosition 647 */ format(BigDecimal number, StringBuffer result, FieldPosition fieldPosition)648 private StringBuffer format(BigDecimal number, StringBuffer result, 649 FieldPosition fieldPosition) { 650 FieldPosition icuFieldPosition = getIcuFieldPosition(fieldPosition); 651 icuDecimalFormat.format(number, result, fieldPosition); 652 fieldPosition.setBeginIndex(icuFieldPosition.getBeginIndex()); 653 fieldPosition.setEndIndex(icuFieldPosition.getEndIndex()); 654 return result; 655 } 656 657 /** 658 * Format a BigInteger to produce a string. 659 * @param number The BigInteger to format 660 * @param result where the text is to be appended 661 * @param fieldPosition On input: an alignment field, if desired. 662 * On output: the offsets of the alignment field. 663 * @return The formatted number string 664 * @exception ArithmeticException if rounding is needed with rounding 665 * mode being set to RoundingMode.UNNECESSARY 666 * @see java.text.FieldPosition 667 */ format(BigInteger number, StringBuffer result, FieldPosition fieldPosition)668 private StringBuffer format(BigInteger number, StringBuffer result, 669 FieldPosition fieldPosition) { 670 FieldPosition icuFieldPosition = getIcuFieldPosition(fieldPosition); 671 icuDecimalFormat.format(number, result, fieldPosition); 672 fieldPosition.setBeginIndex(icuFieldPosition.getBeginIndex()); 673 fieldPosition.setEndIndex(icuFieldPosition.getEndIndex()); 674 return result; 675 } 676 677 /** 678 * Formats an Object producing an <code>AttributedCharacterIterator</code>. 679 * You can use the returned <code>AttributedCharacterIterator</code> 680 * to build the resulting String, as well as to determine information 681 * about the resulting String. 682 * <p> 683 * Each attribute key of the AttributedCharacterIterator will be of type 684 * <code>NumberFormat.Field</code>, with the attribute value being the 685 * same as the attribute key. 686 * 687 * @exception NullPointerException if obj is null. 688 * @exception IllegalArgumentException when the Format cannot format the 689 * given object. 690 * @exception ArithmeticException if rounding is needed with rounding 691 * mode being set to RoundingMode.UNNECESSARY 692 * @param obj The object to format 693 * @return AttributedCharacterIterator describing the formatted value. 694 * @since 1.4 695 */ formatToCharacterIterator(Object obj)696 public AttributedCharacterIterator formatToCharacterIterator(Object obj) { 697 if (obj == null) { 698 throw new NullPointerException("object == null"); 699 } 700 // Note: formatToCharacterIterator cannot be used directly because it returns attributes 701 // in terms of its own class: icu.text.NumberFormat instead of java.text.NumberFormat. 702 // http://bugs.icu-project.org/trac/ticket/11931 Proposes to use the NumberFormat constants. 703 704 AttributedCharacterIterator original = icuDecimalFormat.formatToCharacterIterator(obj); 705 706 // Extract the text out of the ICU iterator. 707 StringBuilder textBuilder = new StringBuilder( 708 original.getEndIndex() - original.getBeginIndex()); 709 710 for (int i = original.getBeginIndex(); i < original.getEndIndex(); i++) { 711 textBuilder.append(original.current()); 712 original.next(); 713 } 714 715 AttributedString result = new AttributedString(textBuilder.toString()); 716 717 for (int i = original.getBeginIndex(); i < original.getEndIndex(); i++) { 718 original.setIndex(i); 719 720 for (AttributedCharacterIterator.Attribute attribute 721 : original.getAttributes().keySet()) { 722 int start = original.getRunStart(); 723 int end = original.getRunLimit(); 724 Field javaAttr = toJavaFieldAttribute(attribute); 725 result.addAttribute(javaAttr, javaAttr, start, end); 726 } 727 } 728 729 return result.getIterator(); 730 } 731 732 /** 733 * Parses text from a string to produce a <code>Number</code>. 734 * <p> 735 * The method attempts to parse text starting at the index given by 736 * <code>pos</code>. 737 * If parsing succeeds, then the index of <code>pos</code> is updated 738 * to the index after the last character used (parsing does not necessarily 739 * use all characters up to the end of the string), and the parsed 740 * number is returned. The updated <code>pos</code> can be used to 741 * indicate the starting point for the next call to this method. 742 * If an error occurs, then the index of <code>pos</code> is not 743 * changed, the error index of <code>pos</code> is set to the index of 744 * the character where the error occurred, and null is returned. 745 * <p> 746 * The subclass returned depends on the value of {@link #isParseBigDecimal} 747 * as well as on the string being parsed. 748 * <ul> 749 * <li>If <code>isParseBigDecimal()</code> is false (the default), 750 * most integer values are returned as <code>Long</code> 751 * objects, no matter how they are written: <code>"17"</code> and 752 * <code>"17.000"</code> both parse to <code>Long(17)</code>. 753 * Values that cannot fit into a <code>Long</code> are returned as 754 * <code>Double</code>s. This includes values with a fractional part, 755 * infinite values, <code>NaN</code>, and the value -0.0. 756 * <code>DecimalFormat</code> does <em>not</em> decide whether to 757 * return a <code>Double</code> or a <code>Long</code> based on the 758 * presence of a decimal separator in the source string. Doing so 759 * would prevent integers that overflow the mantissa of a double, 760 * such as <code>"-9,223,372,036,854,775,808.00"</code>, from being 761 * parsed accurately. 762 * <p> 763 * Callers may use the <code>Number</code> methods 764 * <code>doubleValue</code>, <code>longValue</code>, etc., to obtain 765 * the type they want. 766 * <li>If <code>isParseBigDecimal()</code> is true, values are returned 767 * as <code>BigDecimal</code> objects. The values are the ones 768 * constructed by {@link java.math.BigDecimal#BigDecimal(String)} 769 * for corresponding strings in locale-independent format. The 770 * special cases negative and positive infinity and NaN are returned 771 * as <code>Double</code> instances holding the values of the 772 * corresponding <code>Double</code> constants. 773 * </ul> 774 * <p> 775 * <code>DecimalFormat</code> parses all Unicode characters that represent 776 * decimal digits, as defined by <code>Character.digit()</code>. In 777 * addition, <code>DecimalFormat</code> also recognizes as digits the ten 778 * consecutive characters starting with the localized zero digit defined in 779 * the <code>DecimalFormatSymbols</code> object. 780 * 781 * @param text the string to be parsed 782 * @param pos A <code>ParsePosition</code> object with index and error 783 * index information as described above. 784 * @return the parsed value, or <code>null</code> if the parse fails 785 * @exception NullPointerException if <code>text</code> or 786 * <code>pos</code> is null. 787 */ parse(String text, ParsePosition pos)788 public Number parse(String text, ParsePosition pos) { 789 // Return early if the parse position is bogus. 790 if (pos.index < 0 || pos.index >= text.length()) { 791 return null; 792 } 793 794 // This might return android.icu.math.BigDecimal, java.math.BigInteger or a primitive type. 795 Number number = icuDecimalFormat.parse(text, pos); 796 if (number == null) { 797 return null; 798 } 799 if (isParseBigDecimal()) { 800 if (number instanceof Long) { 801 return new BigDecimal(number.longValue()); 802 } 803 if ((number instanceof Double) && !((Double) number).isInfinite() 804 && !((Double) number).isNaN()) { 805 return new BigDecimal(number.toString()); 806 } 807 if ((number instanceof Double) && 808 (((Double) number).isNaN() || ((Double) number).isInfinite())) { 809 return number; 810 } 811 if (number instanceof android.icu.math.BigDecimal) { 812 return ((android.icu.math.BigDecimal) number).toBigDecimal(); 813 } 814 } 815 if ((number instanceof android.icu.math.BigDecimal) || (number instanceof BigInteger)) { 816 return number.doubleValue(); 817 } 818 if (isParseIntegerOnly() && number.equals(new Double(-0.0))) { 819 return 0L; 820 } 821 return number; 822 } 823 824 /** 825 * Returns a copy of the decimal format symbols, which is generally not 826 * changed by the programmer or user. 827 * @return a copy of the desired DecimalFormatSymbols 828 * @see java.text.DecimalFormatSymbols 829 */ getDecimalFormatSymbols()830 public DecimalFormatSymbols getDecimalFormatSymbols() { 831 return DecimalFormatSymbols.fromIcuInstance(icuDecimalFormat.getDecimalFormatSymbols()); 832 } 833 834 835 /** 836 * Sets the decimal format symbols, which is generally not changed 837 * by the programmer or user. 838 * @param newSymbols desired DecimalFormatSymbols 839 * @see java.text.DecimalFormatSymbols 840 */ setDecimalFormatSymbols(DecimalFormatSymbols newSymbols)841 public void setDecimalFormatSymbols(DecimalFormatSymbols newSymbols) { 842 try { 843 // don't allow multiple references 844 symbols = (DecimalFormatSymbols) newSymbols.clone(); 845 icuDecimalFormat.setDecimalFormatSymbols(symbols.getIcuDecimalFormatSymbols()); 846 } catch (Exception foo) { 847 // should never happen 848 } 849 } 850 851 /** 852 * Get the positive prefix. 853 * <P>Examples: +123, $123, sFr123 854 */ getPositivePrefix()855 public String getPositivePrefix () { 856 return icuDecimalFormat.getPositivePrefix(); 857 } 858 859 /** 860 * Set the positive prefix. 861 * <P>Examples: +123, $123, sFr123 862 */ setPositivePrefix(String newValue)863 public void setPositivePrefix (String newValue) { 864 icuDecimalFormat.setPositivePrefix(newValue); 865 } 866 867 /** 868 * Get the prefix. 869 * <P>Examples: -123, ($123) (with negative suffix), sFr-123 870 */ getNegativePrefix()871 public String getNegativePrefix () { 872 return icuDecimalFormat.getNegativePrefix(); 873 } 874 875 /** 876 * Set the negative prefix. 877 * <P>Examples: -123, ($123) (with negative suffix), sFr-123 878 */ setNegativePrefix(String newValue)879 public void setNegativePrefix (String newValue) { 880 icuDecimalFormat.setNegativePrefix(newValue); 881 } 882 883 /** 884 * Get the positive suffix. 885 * <P>Example: 123% 886 */ getPositiveSuffix()887 public String getPositiveSuffix () { 888 return icuDecimalFormat.getPositiveSuffix(); 889 } 890 891 /** 892 * Set the positive suffix. 893 * <P>Example: 123% 894 */ setPositiveSuffix(String newValue)895 public void setPositiveSuffix (String newValue) { 896 icuDecimalFormat.setPositiveSuffix(newValue); 897 } 898 899 /** 900 * Get the negative suffix. 901 * <P>Examples: -123%, ($123) (with positive suffixes) 902 */ getNegativeSuffix()903 public String getNegativeSuffix () { 904 return icuDecimalFormat.getNegativeSuffix(); 905 } 906 907 /** 908 * Set the negative suffix. 909 * <P>Examples: 123% 910 */ setNegativeSuffix(String newValue)911 public void setNegativeSuffix (String newValue) { 912 icuDecimalFormat.setNegativeSuffix(newValue); 913 } 914 915 /** 916 * Gets the multiplier for use in percent, per mille, and similar 917 * formats. 918 * 919 * @see #setMultiplier(int) 920 */ getMultiplier()921 public int getMultiplier () { 922 return icuDecimalFormat.getMultiplier(); 923 } 924 925 /** 926 * Sets the multiplier for use in percent, per mille, and similar 927 * formats. 928 * For a percent format, set the multiplier to 100 and the suffixes to 929 * have '%' (for Arabic, use the Arabic percent sign). 930 * For a per mille format, set the multiplier to 1000 and the suffixes to 931 * have '\u2030'. 932 * 933 * <P>Example: with multiplier 100, 1.23 is formatted as "123", and 934 * "123" is parsed into 1.23. 935 * 936 * @see #getMultiplier 937 */ setMultiplier(int newValue)938 public void setMultiplier (int newValue) { 939 icuDecimalFormat.setMultiplier(newValue); 940 } 941 942 /** 943 * Return the grouping size. Grouping size is the number of digits between 944 * grouping separators in the integer portion of a number. For example, 945 * in the number "123,456.78", the grouping size is 3. 946 * @see #setGroupingSize 947 * @see java.text.NumberFormat#isGroupingUsed 948 * @see java.text.DecimalFormatSymbols#getGroupingSeparator 949 */ getGroupingSize()950 public int getGroupingSize () { 951 return icuDecimalFormat.getGroupingSize(); 952 } 953 954 /** 955 * Set the grouping size. Grouping size is the number of digits between 956 * grouping separators in the integer portion of a number. For example, 957 * in the number "123,456.78", the grouping size is 3. 958 * <br> 959 * The value passed in is converted to a byte, which may lose information. 960 * @see #getGroupingSize 961 * @see java.text.NumberFormat#setGroupingUsed 962 * @see java.text.DecimalFormatSymbols#setGroupingSeparator 963 */ setGroupingSize(int newValue)964 public void setGroupingSize (int newValue) { 965 icuDecimalFormat.setGroupingSize(newValue); 966 } 967 968 /** 969 * Returns true if grouping is used in this format. For example, in the 970 * English locale, with grouping on, the number 1234567 might be formatted 971 * as "1,234,567". The grouping separator as well as the size of each group 972 * is locale dependant and is determined by sub-classes of NumberFormat. 973 * @see #setGroupingUsed 974 */ isGroupingUsed()975 public boolean isGroupingUsed() { 976 return icuDecimalFormat.isGroupingUsed(); 977 } 978 979 /** 980 * Set whether or not grouping will be used in this format. 981 * @see #isGroupingUsed 982 */ setGroupingUsed(boolean newValue)983 public void setGroupingUsed(boolean newValue) { 984 icuDecimalFormat.setGroupingUsed(newValue); 985 } 986 987 /** 988 * Allows you to get the behavior of the decimal separator with integers. 989 * (The decimal separator will always appear with decimals.) 990 * <P>Example: Decimal ON: 12345 -> 12345.; OFF: 12345 -> 12345 991 */ isDecimalSeparatorAlwaysShown()992 public boolean isDecimalSeparatorAlwaysShown() { 993 return icuDecimalFormat.isDecimalSeparatorAlwaysShown(); 994 } 995 996 /** 997 * Allows you to set the behavior of the decimal separator with integers. 998 * (The decimal separator will always appear with decimals.) 999 * <P>Example: Decimal ON: 12345 -> 12345.; OFF: 12345 -> 12345 1000 */ setDecimalSeparatorAlwaysShown(boolean newValue)1001 public void setDecimalSeparatorAlwaysShown(boolean newValue) { 1002 icuDecimalFormat.setDecimalSeparatorAlwaysShown(newValue); 1003 } 1004 1005 /** 1006 * Returns whether the {@link #parse(java.lang.String, java.text.ParsePosition)} 1007 * method returns <code>BigDecimal</code>. The default value is false. 1008 * @see #setParseBigDecimal 1009 * @since 1.5 1010 */ isParseBigDecimal()1011 public boolean isParseBigDecimal() { 1012 return icuDecimalFormat.isParseBigDecimal(); 1013 } 1014 1015 /** 1016 * Sets whether the {@link #parse(java.lang.String, java.text.ParsePosition)} 1017 * method returns <code>BigDecimal</code>. 1018 * @see #isParseBigDecimal 1019 * @since 1.5 1020 */ setParseBigDecimal(boolean newValue)1021 public void setParseBigDecimal(boolean newValue) { 1022 icuDecimalFormat.setParseBigDecimal(newValue); 1023 } 1024 1025 /** 1026 * Sets whether or not numbers should be parsed as integers only. 1027 * @see #isParseIntegerOnly 1028 */ setParseIntegerOnly(boolean value)1029 public void setParseIntegerOnly(boolean value) { 1030 super.setParseIntegerOnly(value); 1031 icuDecimalFormat.setParseIntegerOnly(value); 1032 } 1033 1034 /** 1035 * Returns true if this format will parse numbers as integers only. 1036 * For example in the English locale, with ParseIntegerOnly true, the 1037 * string "1234." would be parsed as the integer value 1234 and parsing 1038 * would stop at the "." character. Of course, the exact format accepted 1039 * by the parse operation is locale dependant and determined by sub-classes 1040 * of NumberFormat. 1041 */ isParseIntegerOnly()1042 public boolean isParseIntegerOnly() { 1043 return icuDecimalFormat.isParseIntegerOnly(); 1044 } 1045 1046 /** 1047 * Standard override; no change in semantics. 1048 */ clone()1049 public Object clone() { 1050 try { 1051 DecimalFormat other = (DecimalFormat) super.clone(); 1052 other.icuDecimalFormat = (android.icu.text.DecimalFormat) icuDecimalFormat.clone(); 1053 other.symbols = (DecimalFormatSymbols) symbols.clone(); 1054 return other; 1055 } catch (Exception e) { 1056 throw new InternalError(); 1057 } 1058 } 1059 1060 /** 1061 * Overrides equals 1062 */ equals(Object obj)1063 public boolean equals(Object obj) 1064 { 1065 if (obj == null) { 1066 return false; 1067 } 1068 if (this == obj) { 1069 return true; 1070 } 1071 if (!(obj instanceof DecimalFormat)) { 1072 return false; 1073 } 1074 DecimalFormat other = (DecimalFormat) obj; 1075 return icuDecimalFormat.equals(other.icuDecimalFormat) 1076 && compareIcuRoundingIncrement(other.icuDecimalFormat); 1077 } 1078 compareIcuRoundingIncrement(android.icu.text.DecimalFormat other)1079 private boolean compareIcuRoundingIncrement(android.icu.text.DecimalFormat other) { 1080 BigDecimal increment = this.icuDecimalFormat.getRoundingIncrement(); 1081 if (increment != null) { 1082 return (other.getRoundingIncrement() != null) 1083 && increment.equals(other.getRoundingIncrement()); 1084 } 1085 return other.getRoundingIncrement() == null; 1086 } 1087 1088 /** 1089 * Overrides hashCode 1090 */ hashCode()1091 public int hashCode() { 1092 return super.hashCode() * 37 + getPositivePrefix().hashCode(); 1093 // just enough fields for a reasonable distribution 1094 } 1095 1096 /** 1097 * Synthesizes a pattern string that represents the current state 1098 * of this Format object. 1099 * @see #applyPattern 1100 */ toPattern()1101 public String toPattern() { 1102 return icuDecimalFormat.toPattern(); 1103 } 1104 1105 /** 1106 * Synthesizes a localized pattern string that represents the current 1107 * state of this Format object. 1108 * @see #applyPattern 1109 */ toLocalizedPattern()1110 public String toLocalizedPattern() { 1111 return icuDecimalFormat.toLocalizedPattern(); 1112 } 1113 1114 /** 1115 * Apply the given pattern to this Format object. A pattern is a 1116 * short-hand specification for the various formatting properties. 1117 * These properties can also be changed individually through the 1118 * various setter methods. 1119 * <p> 1120 * There is no limit to integer digits set 1121 * by this routine, since that is the typical end-user desire; 1122 * use setMaximumInteger if you want to set a real value. 1123 * For negative numbers, use a second pattern, separated by a semicolon 1124 * <P>Example <code>"#,#00.0#"</code> -> 1,234.56 1125 * <P>This means a minimum of 2 integer digits, 1 fraction digit, and 1126 * a maximum of 2 fraction digits. 1127 * <p>Example: <code>"#,#00.0#;(#,#00.0#)"</code> for negatives in 1128 * parentheses. 1129 * <p>In negative patterns, the minimum and maximum counts are ignored; 1130 * these are presumed to be set in the positive pattern. 1131 * 1132 * @exception NullPointerException if <code>pattern</code> is null 1133 * @exception IllegalArgumentException if the given pattern is invalid. 1134 */ applyPattern(String pattern)1135 public void applyPattern(String pattern) { 1136 icuDecimalFormat.applyPattern(pattern); 1137 updateFieldsFromIcu(); 1138 } 1139 1140 1141 /** 1142 * Apply the given pattern to this Format object. The pattern 1143 * is assumed to be in a localized notation. A pattern is a 1144 * short-hand specification for the various formatting properties. 1145 * These properties can also be changed individually through the 1146 * various setter methods. 1147 * <p> 1148 * There is no limit to integer digits set 1149 * by this routine, since that is the typical end-user desire; 1150 * use setMaximumInteger if you want to set a real value. 1151 * For negative numbers, use a second pattern, separated by a semicolon 1152 * <P>Example <code>"#,#00.0#"</code> -> 1,234.56 1153 * <P>This means a minimum of 2 integer digits, 1 fraction digit, and 1154 * a maximum of 2 fraction digits. 1155 * <p>Example: <code>"#,#00.0#;(#,#00.0#)"</code> for negatives in 1156 * parentheses. 1157 * <p>In negative patterns, the minimum and maximum counts are ignored; 1158 * these are presumed to be set in the positive pattern. 1159 * 1160 * @exception NullPointerException if <code>pattern</code> is null 1161 * @exception IllegalArgumentException if the given pattern is invalid. 1162 */ applyLocalizedPattern(String pattern)1163 public void applyLocalizedPattern(String pattern) { 1164 icuDecimalFormat.applyLocalizedPattern(pattern); 1165 updateFieldsFromIcu(); 1166 } 1167 updateFieldsFromIcu()1168 private void updateFieldsFromIcu() { 1169 // Imitate behaviour of ICU4C NumberFormat that Android used up to M. 1170 // If the pattern doesn't enforce a different value (some exponential 1171 // patterns do), then set the maximum integer digits to 2 billion. 1172 if (icuDecimalFormat.getMaximumIntegerDigits() == DOUBLE_INTEGER_DIGITS) { 1173 icuDecimalFormat.setMaximumIntegerDigits(2000000000); 1174 } 1175 maximumIntegerDigits = icuDecimalFormat.getMaximumIntegerDigits(); 1176 minimumIntegerDigits = icuDecimalFormat.getMinimumIntegerDigits(); 1177 maximumFractionDigits = icuDecimalFormat.getMaximumFractionDigits(); 1178 minimumFractionDigits = icuDecimalFormat.getMinimumFractionDigits(); 1179 } 1180 1181 /** 1182 * Sets the maximum number of digits allowed in the integer portion of a 1183 * number. 1184 * For formatting numbers other than <code>BigInteger</code> and 1185 * <code>BigDecimal</code> objects, the lower of <code>newValue</code> and 1186 * 309 is used. Negative input values are replaced with 0. 1187 * @see NumberFormat#setMaximumIntegerDigits 1188 */ setMaximumIntegerDigits(int newValue)1189 public void setMaximumIntegerDigits(int newValue) { 1190 maximumIntegerDigits = Math.min(Math.max(0, newValue), MAXIMUM_INTEGER_DIGITS); 1191 super.setMaximumIntegerDigits((maximumIntegerDigits > DOUBLE_INTEGER_DIGITS) ? 1192 DOUBLE_INTEGER_DIGITS : maximumIntegerDigits); 1193 if (minimumIntegerDigits > maximumIntegerDigits) { 1194 minimumIntegerDigits = maximumIntegerDigits; 1195 super.setMinimumIntegerDigits((minimumIntegerDigits > DOUBLE_INTEGER_DIGITS) ? 1196 DOUBLE_INTEGER_DIGITS : minimumIntegerDigits); 1197 } 1198 icuDecimalFormat.setMaximumIntegerDigits(getMaximumIntegerDigits()); 1199 } 1200 1201 /** 1202 * Sets the minimum number of digits allowed in the integer portion of a 1203 * number. 1204 * For formatting numbers other than <code>BigInteger</code> and 1205 * <code>BigDecimal</code> objects, the lower of <code>newValue</code> and 1206 * 309 is used. Negative input values are replaced with 0. 1207 * @see NumberFormat#setMinimumIntegerDigits 1208 */ setMinimumIntegerDigits(int newValue)1209 public void setMinimumIntegerDigits(int newValue) { 1210 minimumIntegerDigits = Math.min(Math.max(0, newValue), MAXIMUM_INTEGER_DIGITS); 1211 super.setMinimumIntegerDigits((minimumIntegerDigits > DOUBLE_INTEGER_DIGITS) ? 1212 DOUBLE_INTEGER_DIGITS : minimumIntegerDigits); 1213 if (minimumIntegerDigits > maximumIntegerDigits) { 1214 maximumIntegerDigits = minimumIntegerDigits; 1215 super.setMaximumIntegerDigits((maximumIntegerDigits > DOUBLE_INTEGER_DIGITS) ? 1216 DOUBLE_INTEGER_DIGITS : maximumIntegerDigits); 1217 } 1218 icuDecimalFormat.setMinimumIntegerDigits(getMinimumIntegerDigits()); 1219 } 1220 1221 /** 1222 * Sets the maximum number of digits allowed in the fraction portion of a 1223 * number. 1224 * For formatting numbers other than <code>BigInteger</code> and 1225 * <code>BigDecimal</code> objects, the lower of <code>newValue</code> and 1226 * 340 is used. Negative input values are replaced with 0. 1227 * @see NumberFormat#setMaximumFractionDigits 1228 */ setMaximumFractionDigits(int newValue)1229 public void setMaximumFractionDigits(int newValue) { 1230 maximumFractionDigits = Math.min(Math.max(0, newValue), MAXIMUM_FRACTION_DIGITS); 1231 super.setMaximumFractionDigits((maximumFractionDigits > DOUBLE_FRACTION_DIGITS) ? 1232 DOUBLE_FRACTION_DIGITS : maximumFractionDigits); 1233 if (minimumFractionDigits > maximumFractionDigits) { 1234 minimumFractionDigits = maximumFractionDigits; 1235 super.setMinimumFractionDigits((minimumFractionDigits > DOUBLE_FRACTION_DIGITS) ? 1236 DOUBLE_FRACTION_DIGITS : minimumFractionDigits); 1237 } 1238 icuDecimalFormat.setMaximumFractionDigits(getMaximumFractionDigits()); 1239 } 1240 1241 /** 1242 * Sets the minimum number of digits allowed in the fraction portion of a 1243 * number. 1244 * For formatting numbers other than <code>BigInteger</code> and 1245 * <code>BigDecimal</code> objects, the lower of <code>newValue</code> and 1246 * 340 is used. Negative input values are replaced with 0. 1247 * @see NumberFormat#setMinimumFractionDigits 1248 */ setMinimumFractionDigits(int newValue)1249 public void setMinimumFractionDigits(int newValue) { 1250 minimumFractionDigits = Math.min(Math.max(0, newValue), MAXIMUM_FRACTION_DIGITS); 1251 super.setMinimumFractionDigits((minimumFractionDigits > DOUBLE_FRACTION_DIGITS) ? 1252 DOUBLE_FRACTION_DIGITS : minimumFractionDigits); 1253 if (minimumFractionDigits > maximumFractionDigits) { 1254 maximumFractionDigits = minimumFractionDigits; 1255 super.setMaximumFractionDigits((maximumFractionDigits > DOUBLE_FRACTION_DIGITS) ? 1256 DOUBLE_FRACTION_DIGITS : maximumFractionDigits); 1257 } 1258 icuDecimalFormat.setMinimumFractionDigits(getMinimumFractionDigits()); 1259 } 1260 1261 /** 1262 * Gets the maximum number of digits allowed in the integer portion of a 1263 * number. 1264 * For formatting numbers other than <code>BigInteger</code> and 1265 * <code>BigDecimal</code> objects, the lower of the return value and 1266 * 309 is used. 1267 * @see #setMaximumIntegerDigits 1268 */ getMaximumIntegerDigits()1269 public int getMaximumIntegerDigits() { 1270 return maximumIntegerDigits; 1271 } 1272 1273 /** 1274 * Gets the minimum number of digits allowed in the integer portion of a 1275 * number. 1276 * For formatting numbers other than <code>BigInteger</code> and 1277 * <code>BigDecimal</code> objects, the lower of the return value and 1278 * 309 is used. 1279 * @see #setMinimumIntegerDigits 1280 */ getMinimumIntegerDigits()1281 public int getMinimumIntegerDigits() { 1282 return minimumIntegerDigits; 1283 } 1284 1285 /** 1286 * Gets the maximum number of digits allowed in the fraction portion of a 1287 * number. 1288 * For formatting numbers other than <code>BigInteger</code> and 1289 * <code>BigDecimal</code> objects, the lower of the return value and 1290 * 340 is used. 1291 * @see #setMaximumFractionDigits 1292 */ getMaximumFractionDigits()1293 public int getMaximumFractionDigits() { 1294 return maximumFractionDigits; 1295 } 1296 1297 /** 1298 * Gets the minimum number of digits allowed in the fraction portion of a 1299 * number. 1300 * For formatting numbers other than <code>BigInteger</code> and 1301 * <code>BigDecimal</code> objects, the lower of the return value and 1302 * 340 is used. 1303 * @see #setMinimumFractionDigits 1304 */ getMinimumFractionDigits()1305 public int getMinimumFractionDigits() { 1306 return minimumFractionDigits; 1307 } 1308 1309 /** 1310 * Gets the currency used by this decimal format when formatting 1311 * currency values. 1312 * The currency is obtained by calling 1313 * {@link DecimalFormatSymbols#getCurrency DecimalFormatSymbols.getCurrency} 1314 * on this number format's symbols. 1315 * 1316 * @return the currency used by this decimal format, or <code>null</code> 1317 * @since 1.4 1318 */ getCurrency()1319 public Currency getCurrency() { 1320 return symbols.getCurrency(); 1321 } 1322 1323 /** 1324 * Sets the currency used by this number format when formatting 1325 * currency values. This does not update the minimum or maximum 1326 * number of fraction digits used by the number format. 1327 * The currency is set by calling 1328 * {@link DecimalFormatSymbols#setCurrency DecimalFormatSymbols.setCurrency} 1329 * on this number format's symbols. 1330 * 1331 * @param currency the new currency to be used by this decimal format 1332 * @exception NullPointerException if <code>currency</code> is null 1333 * @since 1.4 1334 */ setCurrency(Currency currency)1335 public void setCurrency(Currency currency) { 1336 // Set the international currency symbol, and currency symbol on the DecimalFormatSymbols 1337 // object and tell ICU to use that. 1338 if (currency != symbols.getCurrency() 1339 || !currency.getSymbol().equals(symbols.getCurrencySymbol())) { 1340 symbols.setCurrency(currency); 1341 icuDecimalFormat.setDecimalFormatSymbols(symbols.getIcuDecimalFormatSymbols()); 1342 // Giving the icuDecimalFormat a new currency will cause the fractional digits to be 1343 // updated. This class is specified to not touch the fraction digits, so we re-set them. 1344 icuDecimalFormat.setMinimumFractionDigits(minimumFractionDigits); 1345 icuDecimalFormat.setMaximumFractionDigits(maximumFractionDigits); 1346 } 1347 } 1348 1349 /** 1350 * Gets the {@link java.math.RoundingMode} used in this DecimalFormat. 1351 * 1352 * @return The <code>RoundingMode</code> used for this DecimalFormat. 1353 * @see #setRoundingMode(RoundingMode) 1354 * @since 1.6 1355 */ getRoundingMode()1356 public RoundingMode getRoundingMode() { 1357 return roundingMode; 1358 } 1359 convertRoundingMode(RoundingMode rm)1360 private static int convertRoundingMode(RoundingMode rm) { 1361 switch (rm) { 1362 case UP: 1363 return MathContext.ROUND_UP; 1364 case DOWN: 1365 return MathContext.ROUND_DOWN; 1366 case CEILING: 1367 return MathContext.ROUND_CEILING; 1368 case FLOOR: 1369 return MathContext.ROUND_FLOOR; 1370 case HALF_UP: 1371 return MathContext.ROUND_HALF_UP; 1372 case HALF_DOWN: 1373 return MathContext.ROUND_HALF_DOWN; 1374 case HALF_EVEN: 1375 return MathContext.ROUND_HALF_EVEN; 1376 case UNNECESSARY: 1377 return MathContext.ROUND_UNNECESSARY; 1378 } 1379 throw new IllegalArgumentException("Invalid rounding mode specified"); 1380 } 1381 1382 /** 1383 * Sets the {@link java.math.RoundingMode} used in this DecimalFormat. 1384 * 1385 * @param roundingMode The <code>RoundingMode</code> to be used 1386 * @see #getRoundingMode() 1387 * @exception NullPointerException if <code>roundingMode</code> is null. 1388 * @since 1.6 1389 */ setRoundingMode(RoundingMode roundingMode)1390 public void setRoundingMode(RoundingMode roundingMode) { 1391 if (roundingMode == null) { 1392 throw new NullPointerException(); 1393 } 1394 1395 this.roundingMode = roundingMode; 1396 1397 icuDecimalFormat.setRoundingMode(convertRoundingMode(roundingMode)); 1398 } 1399 1400 /** 1401 * Adjusts the minimum and maximum fraction digits to values that 1402 * are reasonable for the currency's default fraction digits. 1403 */ adjustForCurrencyDefaultFractionDigits()1404 void adjustForCurrencyDefaultFractionDigits() { 1405 Currency currency = symbols.getCurrency(); 1406 if (currency == null) { 1407 try { 1408 currency = Currency.getInstance(symbols.getInternationalCurrencySymbol()); 1409 } catch (IllegalArgumentException e) { 1410 } 1411 } 1412 if (currency != null) { 1413 int digits = currency.getDefaultFractionDigits(); 1414 if (digits != -1) { 1415 int oldMinDigits = getMinimumFractionDigits(); 1416 // Common patterns are "#.##", "#.00", "#". 1417 // Try to adjust all of them in a reasonable way. 1418 if (oldMinDigits == getMaximumFractionDigits()) { 1419 setMinimumFractionDigits(digits); 1420 setMaximumFractionDigits(digits); 1421 } else { 1422 setMinimumFractionDigits(Math.min(digits, oldMinDigits)); 1423 setMaximumFractionDigits(digits); 1424 } 1425 } 1426 } 1427 } 1428 1429 private static final int currentSerialVersion = 4; 1430 1431 // the fields list to be serialized 1432 private static final ObjectStreamField[] serialPersistentFields = { 1433 new ObjectStreamField("positivePrefix", String.class), 1434 new ObjectStreamField("positiveSuffix", String.class), 1435 new ObjectStreamField("negativePrefix", String.class), 1436 new ObjectStreamField("negativeSuffix", String.class), 1437 new ObjectStreamField("posPrefixPattern", String.class), 1438 new ObjectStreamField("posSuffixPattern", String.class), 1439 new ObjectStreamField("negPrefixPattern", String.class), 1440 new ObjectStreamField("negSuffixPattern", String.class), 1441 new ObjectStreamField("multiplier", int.class), 1442 new ObjectStreamField("groupingSize", byte.class), 1443 new ObjectStreamField("groupingUsed", boolean.class), 1444 new ObjectStreamField("decimalSeparatorAlwaysShown", boolean.class), 1445 new ObjectStreamField("parseBigDecimal", boolean.class), 1446 new ObjectStreamField("roundingMode", RoundingMode.class), 1447 new ObjectStreamField("symbols", DecimalFormatSymbols.class), 1448 new ObjectStreamField("useExponentialNotation", boolean.class), 1449 new ObjectStreamField("minExponentDigits", byte.class), 1450 new ObjectStreamField("maximumIntegerDigits", int.class), 1451 new ObjectStreamField("minimumIntegerDigits", int.class), 1452 new ObjectStreamField("maximumFractionDigits", int.class), 1453 new ObjectStreamField("minimumFractionDigits", int.class), 1454 new ObjectStreamField("serialVersionOnStream", int.class), 1455 }; 1456 writeObject(ObjectOutputStream stream)1457 private void writeObject(ObjectOutputStream stream) throws IOException, ClassNotFoundException { 1458 ObjectOutputStream.PutField fields = stream.putFields(); 1459 fields.put("positivePrefix", icuDecimalFormat.getPositivePrefix()); 1460 fields.put("positiveSuffix", icuDecimalFormat.getPositiveSuffix()); 1461 fields.put("negativePrefix", icuDecimalFormat.getNegativePrefix()); 1462 fields.put("negativeSuffix", icuDecimalFormat.getNegativeSuffix()); 1463 fields.put("posPrefixPattern", (String) null); 1464 fields.put("posSuffixPattern", (String) null); 1465 fields.put("negPrefixPattern", (String) null); 1466 fields.put("negSuffixPattern", (String) null); 1467 fields.put("multiplier", icuDecimalFormat.getMultiplier()); 1468 fields.put("groupingSize", (byte) icuDecimalFormat.getGroupingSize()); 1469 fields.put("groupingUsed", icuDecimalFormat.isGroupingUsed()); 1470 fields.put("decimalSeparatorAlwaysShown", icuDecimalFormat.isDecimalSeparatorAlwaysShown()); 1471 fields.put("parseBigDecimal", icuDecimalFormat.isParseBigDecimal()); 1472 fields.put("roundingMode", roundingMode); 1473 fields.put("symbols", symbols); 1474 fields.put("useExponentialNotation", false); 1475 fields.put("minExponentDigits", (byte) 0); 1476 fields.put("maximumIntegerDigits", icuDecimalFormat.getMaximumIntegerDigits()); 1477 fields.put("minimumIntegerDigits", icuDecimalFormat.getMinimumIntegerDigits()); 1478 fields.put("maximumFractionDigits", icuDecimalFormat.getMaximumFractionDigits()); 1479 fields.put("minimumFractionDigits", icuDecimalFormat.getMinimumFractionDigits()); 1480 fields.put("serialVersionOnStream", currentSerialVersion); 1481 stream.writeFields(); 1482 } 1483 1484 /** 1485 * Reads the default serializable fields from the stream and performs 1486 * validations and adjustments for older serialized versions. The 1487 * validations and adjustments are: 1488 * <ol> 1489 * <li> 1490 * Verify that the superclass's digit count fields correctly reflect 1491 * the limits imposed on formatting numbers other than 1492 * <code>BigInteger</code> and <code>BigDecimal</code> objects. These 1493 * limits are stored in the superclass for serialization compatibility 1494 * with older versions, while the limits for <code>BigInteger</code> and 1495 * <code>BigDecimal</code> objects are kept in this class. 1496 * If, in the superclass, the minimum or maximum integer digit count is 1497 * larger than <code>DOUBLE_INTEGER_DIGITS</code> or if the minimum or 1498 * maximum fraction digit count is larger than 1499 * <code>DOUBLE_FRACTION_DIGITS</code>, then the stream data is invalid 1500 * and this method throws an <code>InvalidObjectException</code>. 1501 * <li> 1502 * If <code>serialVersionOnStream</code> is less than 4, initialize 1503 * <code>roundingMode</code> to {@link java.math.RoundingMode#HALF_EVEN 1504 * RoundingMode.HALF_EVEN}. This field is new with version 4. 1505 * <li> 1506 * If <code>serialVersionOnStream</code> is less than 3, then call 1507 * the setters for the minimum and maximum integer and fraction digits with 1508 * the values of the corresponding superclass getters to initialize the 1509 * fields in this class. The fields in this class are new with version 3. 1510 * <li> 1511 * If <code>serialVersionOnStream</code> is less than 1, indicating that 1512 * the stream was written by JDK 1.1, initialize 1513 * <code>useExponentialNotation</code> 1514 * to false, since it was not present in JDK 1.1. 1515 * <li> 1516 * Set <code>serialVersionOnStream</code> to the maximum allowed value so 1517 * that default serialization will work properly if this object is streamed 1518 * out again. 1519 * </ol> 1520 * 1521 * <p>Stream versions older than 2 will not have the affix pattern variables 1522 * <code>posPrefixPattern</code> etc. As a result, they will be initialized 1523 * to <code>null</code>, which means the affix strings will be taken as 1524 * literal values. This is exactly what we want, since that corresponds to 1525 * the pre-version-2 behavior. 1526 */ readObject(ObjectInputStream stream)1527 private void readObject(ObjectInputStream stream) 1528 throws IOException, ClassNotFoundException { 1529 ObjectInputStream.GetField fields = stream.readFields(); 1530 this.symbols = (DecimalFormatSymbols) fields.get("symbols", null); 1531 1532 init(""); 1533 1534 icuDecimalFormat.setPositivePrefix((String) fields.get("positivePrefix", "")); 1535 icuDecimalFormat.setPositiveSuffix((String) fields.get("positiveSuffix", "")); 1536 icuDecimalFormat.setNegativePrefix((String) fields.get("negativePrefix", "-")); 1537 icuDecimalFormat.setNegativeSuffix((String) fields.get("negativeSuffix", "")); 1538 icuDecimalFormat.setMultiplier(fields.get("multiplier", 1)); 1539 icuDecimalFormat.setGroupingSize(fields.get("groupingSize", (byte) 3)); 1540 icuDecimalFormat.setGroupingUsed(fields.get("groupingUsed", true)); 1541 icuDecimalFormat.setDecimalSeparatorAlwaysShown(fields.get("decimalSeparatorAlwaysShown", 1542 false)); 1543 1544 setRoundingMode((RoundingMode) fields.get("roundingMode", RoundingMode.HALF_EVEN)); 1545 1546 final int maximumIntegerDigits = fields.get("maximumIntegerDigits", 309); 1547 final int minimumIntegerDigits = fields.get("minimumIntegerDigits", 309); 1548 final int maximumFractionDigits = fields.get("maximumFractionDigits", 340); 1549 final int minimumFractionDigits = fields.get("minimumFractionDigits", 340); 1550 // Tell ICU what we want, then ask it what we can have, and then 1551 // set that in our Java object. This isn't RI-compatible, but then very little of our 1552 // behavior in this area is, and it's not obvious how we can second-guess ICU (or tell 1553 // it to just do exactly what we ask). We only need to do this with maximumIntegerDigits 1554 // because ICU doesn't seem to have its own ideas about the other options. 1555 icuDecimalFormat.setMaximumIntegerDigits(maximumIntegerDigits); 1556 super.setMaximumIntegerDigits(icuDecimalFormat.getMaximumIntegerDigits()); 1557 1558 setMinimumIntegerDigits(minimumIntegerDigits); 1559 setMinimumFractionDigits(minimumFractionDigits); 1560 setMaximumFractionDigits(maximumFractionDigits); 1561 setParseBigDecimal(fields.get("parseBigDecimal", false)); 1562 1563 if (fields.get("serialVersionOnStream", 0) < 3) { 1564 setMaximumIntegerDigits(super.getMaximumIntegerDigits()); 1565 setMinimumIntegerDigits(super.getMinimumIntegerDigits()); 1566 setMaximumFractionDigits(super.getMaximumFractionDigits()); 1567 setMinimumFractionDigits(super.getMinimumFractionDigits()); 1568 } 1569 } 1570 1571 //---------------------------------------------------------------------- 1572 // INSTANCE VARIABLES 1573 //---------------------------------------------------------------------- 1574 1575 /** 1576 * The <code>DecimalFormatSymbols</code> object used by this format. 1577 * It contains the symbols used to format numbers, e.g. the grouping separator, 1578 * decimal separator, and so on. 1579 * 1580 * @serial 1581 * @see #setDecimalFormatSymbols 1582 * @see java.text.DecimalFormatSymbols 1583 */ 1584 private DecimalFormatSymbols symbols; 1585 1586 /** 1587 * The maximum number of digits allowed in the integer portion of a 1588 * <code>BigInteger</code> or <code>BigDecimal</code> number. 1589 * <code>maximumIntegerDigits</code> must be greater than or equal to 1590 * <code>minimumIntegerDigits</code>. 1591 * 1592 * @serial 1593 * @see #getMaximumIntegerDigits 1594 * @since 1.5 1595 */ 1596 private int maximumIntegerDigits; 1597 1598 /** 1599 * The minimum number of digits allowed in the integer portion of a 1600 * <code>BigInteger</code> or <code>BigDecimal</code> number. 1601 * <code>minimumIntegerDigits</code> must be less than or equal to 1602 * <code>maximumIntegerDigits</code>. 1603 * 1604 * @serial 1605 * @see #getMinimumIntegerDigits 1606 * @since 1.5 1607 */ 1608 private int minimumIntegerDigits; 1609 1610 /** 1611 * The maximum number of digits allowed in the fractional portion of a 1612 * <code>BigInteger</code> or <code>BigDecimal</code> number. 1613 * <code>maximumFractionDigits</code> must be greater than or equal to 1614 * <code>minimumFractionDigits</code>. 1615 * 1616 * @serial 1617 * @see #getMaximumFractionDigits 1618 * @since 1.5 1619 */ 1620 private int maximumFractionDigits; 1621 1622 /** 1623 * The minimum number of digits allowed in the fractional portion of a 1624 * <code>BigInteger</code> or <code>BigDecimal</code> number. 1625 * <code>minimumFractionDigits</code> must be less than or equal to 1626 * <code>maximumFractionDigits</code>. 1627 * 1628 * @serial 1629 * @see #getMinimumFractionDigits 1630 * @since 1.5 1631 */ 1632 private int minimumFractionDigits; 1633 1634 /** 1635 * The {@link java.math.RoundingMode} used in this DecimalFormat. 1636 * 1637 * @serial 1638 * @since 1.6 1639 */ 1640 private RoundingMode roundingMode = RoundingMode.HALF_EVEN; 1641 1642 1643 1644 //---------------------------------------------------------------------- 1645 // CONSTANTS 1646 //---------------------------------------------------------------------- 1647 1648 // Upper limit on integer and fraction digits for a Java double 1649 static final int DOUBLE_INTEGER_DIGITS = 309; 1650 static final int DOUBLE_FRACTION_DIGITS = 340; 1651 1652 // Upper limit on integer and fraction digits for BigDecimal and BigInteger 1653 static final int MAXIMUM_INTEGER_DIGITS = Integer.MAX_VALUE; 1654 static final int MAXIMUM_FRACTION_DIGITS = Integer.MAX_VALUE; 1655 1656 // Proclaim JDK 1.1 serial compatibility. 1657 static final long serialVersionUID = 864413376551465018L; 1658 1659 /** 1660 * Cache to hold the NumberPattern of a Locale. 1661 */ 1662 private static final ConcurrentMap<Locale, String> cachedLocaleData 1663 = new ConcurrentHashMap<Locale, String>(3); 1664 } 1665