41
41
import java .io .InvalidObjectException ;
42
42
import java .io .IOException ;
43
43
import java .io .ObjectInputStream ;
44
+ import java .util .ArrayList ;
44
45
import java .util .Arrays ;
46
+ import java .util .Objects ;
45
47
46
48
/**
47
49
* {@code ChoiceFormat} is a concrete subclass of {@code NumberFormat} that
@@ -238,6 +240,7 @@ public class ChoiceFormat extends NumberFormat {
238
240
* @see #ChoiceFormat(String)
239
241
*/
240
242
public void applyPattern (String newPattern ) {
243
+ Objects .requireNonNull (newPattern , "newPattern must not be null" );
241
244
applyPatternImpl (newPattern );
242
245
}
243
246
@@ -249,86 +252,92 @@ public void applyPattern(String newPattern) {
249
252
* further understanding of certain special characters: "#", "<", "\u2264", "|".
250
253
*/
251
254
private void applyPatternImpl (String newPattern ) {
252
- StringBuilder [] segments = new StringBuilder [2 ];
253
- for (int i = 0 ; i < segments .length ; ++i ) {
254
- segments [i ] = new StringBuilder ();
255
- }
256
- double [] newChoiceLimits = new double [30 ];
257
- String [] newChoiceFormats = new String [30 ];
258
- int count = 0 ;
259
- int part = 0 ; // 0 denotes limit, 1 denotes format
260
- double startValue = 0 ;
261
- double oldStartValue = Double .NaN ;
255
+ // Set up components
256
+ ArrayList <Double > limits = new ArrayList <>();
257
+ ArrayList <String > formats = new ArrayList <>();
258
+ StringBuilder [] segments = new StringBuilder []{new StringBuilder (),
259
+ new StringBuilder ()};
260
+ int part = 0 ; // 0 denotes LIMIT. 1 denotes FORMAT.
261
+ double limit = 0 ;
262
262
boolean inQuote = false ;
263
+
264
+ // Parse the string, alternating the value of part
263
265
for (int i = 0 ; i < newPattern .length (); ++i ) {
264
266
char ch = newPattern .charAt (i );
265
- if (ch =='\'' ) {
266
- // Check for "''" indicating a literal quote
267
- if ((i +1 )<newPattern .length () && newPattern .charAt (i +1 )==ch ) {
267
+ switch (ch ) {
268
+ case '\'' :
269
+ // Check for "''" indicating a literal quote
270
+ if ((i + 1 ) < newPattern .length () && newPattern .charAt (i + 1 ) == ch ) {
271
+ segments [part ].append (ch );
272
+ ++i ;
273
+ } else {
274
+ inQuote = !inQuote ;
275
+ }
276
+ break ;
277
+ case '<' , '#' , '\u2264' :
278
+ if (inQuote || part == 1 ) {
279
+ // Don't interpret relational symbols if parsing the format
280
+ segments [part ].append (ch );
281
+ } else {
282
+ // Build the numerical value of the limit
283
+ // and switch to parsing format
284
+ if (segments [0 ].isEmpty ()) {
285
+ throw new IllegalArgumentException ("Each interval must" +
286
+ " contain a number before a format" );
287
+ }
288
+ limit = stringToNum (segments [0 ].toString ());
289
+ if (ch == '<' && Double .isFinite (limit )) {
290
+ limit = nextDouble (limit );
291
+ }
292
+ if (!limits .isEmpty () && limit <= limits .getLast ()) {
293
+ throw new IllegalArgumentException ("Incorrect order " +
294
+ "of intervals, must be in ascending order" );
295
+ }
296
+ segments [0 ].setLength (0 );
297
+ part = 1 ;
298
+ }
299
+ break ;
300
+ case '|' :
301
+ if (inQuote ) {
302
+ segments [part ].append (ch );
303
+ } else {
304
+ if (part != 1 ) {
305
+ // Discard incorrect portion and finish building cFmt
306
+ break ;
307
+ }
308
+ // Insert an entry into the format and limit arrays
309
+ // and switch to parsing limit
310
+ limits .add (limit );
311
+ formats .add (segments [1 ].toString ());
312
+ segments [1 ].setLength (0 );
313
+ part = 0 ;
314
+ }
315
+ break ;
316
+ default :
268
317
segments [part ].append (ch );
269
- ++i ;
270
- } else {
271
- inQuote = !inQuote ;
272
- }
273
- } else if (inQuote ) {
274
- segments [part ].append (ch );
275
- } else if (part == 0 && (ch == '<' || ch == '#' || ch == '\u2264' )) {
276
- // Only consider relational symbols if parsing the limit segment (part == 0).
277
- // Don't treat a relational symbol as syntactically significant
278
- // when parsing Format segment (part == 1)
279
- if (segments [0 ].length () == 0 ) {
280
- throw new IllegalArgumentException ("Each interval must"
281
- + " contain a number before a format" );
282
- }
283
-
284
- String tempBuffer = segments [0 ].toString ();
285
- if (tempBuffer .equals ("\u221E " )) {
286
- startValue = Double .POSITIVE_INFINITY ;
287
- } else if (tempBuffer .equals ("-\u221E " )) {
288
- startValue = Double .NEGATIVE_INFINITY ;
289
- } else {
290
- startValue = Double .parseDouble (tempBuffer );
291
- }
292
-
293
- if (ch == '<' && startValue != Double .POSITIVE_INFINITY &&
294
- startValue != Double .NEGATIVE_INFINITY ) {
295
- startValue = nextDouble (startValue );
296
- }
297
- if (startValue <= oldStartValue ) {
298
- throw new IllegalArgumentException ("Incorrect order of"
299
- + " intervals, must be in ascending order" );
300
- }
301
- segments [0 ].setLength (0 );
302
- part = 1 ;
303
- } else if (ch == '|' ) {
304
- if (count == newChoiceLimits .length ) {
305
- newChoiceLimits = doubleArraySize (newChoiceLimits );
306
- newChoiceFormats = doubleArraySize (newChoiceFormats );
307
- }
308
- newChoiceLimits [count ] = startValue ;
309
- newChoiceFormats [count ] = segments [1 ].toString ();
310
- ++count ;
311
- oldStartValue = startValue ;
312
- segments [1 ].setLength (0 );
313
- part = 0 ;
314
- } else {
315
- segments [part ].append (ch );
316
318
}
317
319
}
318
- // clean up last one
320
+
321
+ // clean up last one (SubPattern without trailing '|')
319
322
if (part == 1 ) {
320
- if (count == newChoiceLimits .length ) {
321
- newChoiceLimits = doubleArraySize (newChoiceLimits );
322
- newChoiceFormats = doubleArraySize (newChoiceFormats );
323
- }
324
- newChoiceLimits [count ] = startValue ;
325
- newChoiceFormats [count ] = segments [1 ].toString ();
326
- ++count ;
323
+ limits .add (limit );
324
+ formats .add (segments [1 ].toString ());
327
325
}
328
- choiceLimits = new double [count ];
329
- System .arraycopy (newChoiceLimits , 0 , choiceLimits , 0 , count );
330
- choiceFormats = new String [count ];
331
- System .arraycopy (newChoiceFormats , 0 , choiceFormats , 0 , count );
326
+ choiceLimits = limits .stream ().mapToDouble (d -> d ).toArray ();
327
+ choiceFormats = formats .toArray (new String [0 ]);
328
+ }
329
+
330
+ /**
331
+ * Converts a string value to its double representation; this is used
332
+ * to create the limit segment while applying a pattern.
333
+ * Handles "\u221E", as specified by the pattern syntax.
334
+ */
335
+ private static double stringToNum (String str ) {
336
+ return switch (str ) {
337
+ case "\u221E " -> Double .POSITIVE_INFINITY ;
338
+ case "-\u221E " -> Double .NEGATIVE_INFINITY ;
339
+ default -> Double .parseDouble (str );
340
+ };
332
341
}
333
342
334
343
/**
@@ -402,6 +411,7 @@ public String toPattern() {
402
411
* @see #applyPattern
403
412
*/
404
413
public ChoiceFormat (String newPattern ) {
414
+ Objects .requireNonNull (newPattern , "newPattern must not be null" );
405
415
applyPatternImpl (newPattern );
406
416
}
407
417
@@ -574,6 +584,24 @@ public static final double nextDouble (double d) {
574
584
return Math .nextUp (d );
575
585
}
576
586
587
+ /**
588
+ * Finds the least double greater than {@code d} (if {@code positive} is
589
+ * {@code true}), or the greatest double less than {@code d} (if
590
+ * {@code positive} is {@code false}).
591
+ * If {@code NaN}, returns same value.
592
+ *
593
+ * @implNote This is equivalent to calling
594
+ * {@code positive ? Math.nextUp(d) : Math.nextDown(d)}
595
+ *
596
+ * @param d the reference value
597
+ * @param positive {@code true} if the least double is desired;
598
+ * {@code false} otherwise
599
+ * @return the least or greater double value
600
+ */
601
+ public static double nextDouble (double d , boolean positive ) {
602
+ return positive ? Math .nextUp (d ) : Math .nextDown (d );
603
+ }
604
+
577
605
/**
578
606
* Finds the greatest double less than {@code d}.
579
607
* If {@code NaN}, returns same value.
@@ -593,8 +621,7 @@ public static final double previousDouble (double d) {
593
621
* Overrides Cloneable
594
622
*/
595
623
@ Override
596
- public Object clone ()
597
- {
624
+ public Object clone () {
598
625
ChoiceFormat other = (ChoiceFormat ) super .clone ();
599
626
// for primitives or immutables, shallow clone is enough
600
627
other .choiceLimits = choiceLimits .clone ();
@@ -685,37 +712,4 @@ private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundE
685
712
* @serial
686
713
*/
687
714
private String [] choiceFormats ;
688
-
689
- /**
690
- * Finds the least double greater than {@code d} (if {@code positive} is
691
- * {@code true}), or the greatest double less than {@code d} (if
692
- * {@code positive} is {@code false}).
693
- * If {@code NaN}, returns same value.
694
- *
695
- * @implNote This is equivalent to calling
696
- * {@code positive ? Math.nextUp(d) : Math.nextDown(d)}
697
- *
698
- * @param d the reference value
699
- * @param positive {@code true} if the least double is desired;
700
- * {@code false} otherwise
701
- * @return the least or greater double value
702
- */
703
- public static double nextDouble (double d , boolean positive ) {
704
- return positive ? Math .nextUp (d ) : Math .nextDown (d );
705
- }
706
-
707
- private static double [] doubleArraySize (double [] array ) {
708
- int oldSize = array .length ;
709
- double [] newArray = new double [oldSize * 2 ];
710
- System .arraycopy (array , 0 , newArray , 0 , oldSize );
711
- return newArray ;
712
- }
713
-
714
- private String [] doubleArraySize (String [] array ) {
715
- int oldSize = array .length ;
716
- String [] newArray = new String [oldSize * 2 ];
717
- System .arraycopy (array , 0 , newArray , 0 , oldSize );
718
- return newArray ;
719
- }
720
-
721
715
}
0 commit comments