• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 &#92;uFFFE, &#92;uFFFF, and special characters
103  * <i>Suffix:</i>
104  *         any Unicode characters except &#92;uFFFE, &#92;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>&#92;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>&#164;</code> (<code>&#92;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>&#92;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>&#92;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 '&#92;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