@@ -140,14 +140,6 @@ public final class HexFormat {
140
140
// Access to create strings from a byte array.
141
141
private static final JavaLangAccess jla = SharedSecrets .getJavaLangAccess ();
142
142
143
- private static final byte [] UPPERCASE_DIGITS = {
144
- '0' , '1' , '2' , '3' , '4' , '5' , '6' , '7' ,
145
- '8' , '9' , 'A' , 'B' , 'C' , 'D' , 'E' , 'F' ,
146
- };
147
- private static final byte [] LOWERCASE_DIGITS = {
148
- '0' , '1' , '2' , '3' , '4' , '5' , '6' , '7' ,
149
- '8' , '9' , 'a' , 'b' , 'c' , 'd' , 'e' , 'f' ,
150
- };
151
143
// Analysis has shown that generating the whole array allows the JIT to generate
152
144
// better code compared to a slimmed down array, such as one cutting off after 'f'
153
145
private static final byte [] DIGITS = {
@@ -173,29 +165,37 @@ public final class HexFormat {
173
165
* The hexadecimal characters are from lowercase alpha digits.
174
166
*/
175
167
private static final HexFormat HEX_FORMAT =
176
- new HexFormat ("" , "" , "" , LOWERCASE_DIGITS );
168
+ new HexFormat ("" , "" , "" , Case .LOWERCASE );
169
+
170
+ private static final HexFormat HEX_UPPER_FORMAT =
171
+ new HexFormat ("" , "" , "" , Case .UPPERCASE );
177
172
178
173
private static final byte [] EMPTY_BYTES = {};
179
174
180
175
private final String delimiter ;
181
176
private final String prefix ;
182
177
private final String suffix ;
183
- private final byte [] digits ;
178
+ private final Case digitCase ;
179
+
180
+ private enum Case {
181
+ LOWERCASE ,
182
+ UPPERCASE
183
+ }
184
184
185
185
/**
186
186
* Returns a HexFormat with a delimiter, prefix, suffix, and array of digits.
187
187
*
188
188
* @param delimiter a delimiter, non-null
189
189
* @param prefix a prefix, non-null
190
190
* @param suffix a suffix, non-null
191
- * @param digits byte array of digits indexed by low nibble, non-null
191
+ * @param digitCase enum indicating how to case digits
192
192
* @throws NullPointerException if any argument is null
193
193
*/
194
- private HexFormat (String delimiter , String prefix , String suffix , byte [] digits ) {
194
+ private HexFormat (String delimiter , String prefix , String suffix , Case digitCase ) {
195
195
this .delimiter = Objects .requireNonNull (delimiter , "delimiter" );
196
196
this .prefix = Objects .requireNonNull (prefix , "prefix" );
197
197
this .suffix = Objects .requireNonNull (suffix , "suffix" );
198
- this .digits = digits ;
198
+ this .digitCase = digitCase ;
199
199
}
200
200
201
201
/**
@@ -224,7 +224,7 @@ public static HexFormat of() {
224
224
* @return a {@link HexFormat} with the delimiter and lowercase characters
225
225
*/
226
226
public static HexFormat ofDelimiter (String delimiter ) {
227
- return new HexFormat (delimiter , "" , "" , LOWERCASE_DIGITS );
227
+ return new HexFormat (delimiter , "" , "" , Case . LOWERCASE );
228
228
}
229
229
230
230
/**
@@ -233,7 +233,7 @@ public static HexFormat ofDelimiter(String delimiter) {
233
233
* @return a copy of this {@code HexFormat} with the delimiter
234
234
*/
235
235
public HexFormat withDelimiter (String delimiter ) {
236
- return new HexFormat (delimiter , this .prefix , this .suffix , this .digits );
236
+ return new HexFormat (delimiter , this .prefix , this .suffix , this .digitCase );
237
237
}
238
238
239
239
/**
@@ -243,7 +243,7 @@ public HexFormat withDelimiter(String delimiter) {
243
243
* @return a copy of this {@code HexFormat} with the prefix
244
244
*/
245
245
public HexFormat withPrefix (String prefix ) {
246
- return new HexFormat (this .delimiter , prefix , this .suffix , this .digits );
246
+ return new HexFormat (this .delimiter , prefix , this .suffix , this .digitCase );
247
247
}
248
248
249
249
/**
@@ -253,7 +253,7 @@ public HexFormat withPrefix(String prefix) {
253
253
* @return a copy of this {@code HexFormat} with the suffix
254
254
*/
255
255
public HexFormat withSuffix (String suffix ) {
256
- return new HexFormat (this .delimiter , this .prefix , suffix , this .digits );
256
+ return new HexFormat (this .delimiter , this .prefix , suffix , this .digitCase );
257
257
}
258
258
259
259
/**
@@ -263,7 +263,9 @@ public HexFormat withSuffix(String suffix) {
263
263
* @return a copy of this {@code HexFormat} with uppercase hexadecimal characters
264
264
*/
265
265
public HexFormat withUpperCase () {
266
- return new HexFormat (this .delimiter , this .prefix , this .suffix , UPPERCASE_DIGITS );
266
+ if (this == HEX_FORMAT )
267
+ return HEX_UPPER_FORMAT ;
268
+ return new HexFormat (this .delimiter , this .prefix , this .suffix , Case .UPPERCASE );
267
269
}
268
270
269
271
/**
@@ -273,7 +275,7 @@ public HexFormat withUpperCase() {
273
275
* @return a copy of this {@code HexFormat} with lowercase hexadecimal characters
274
276
*/
275
277
public HexFormat withLowerCase () {
276
- return new HexFormat (this .delimiter , this .prefix , this .suffix , LOWERCASE_DIGITS );
278
+ return new HexFormat (this .delimiter , this .prefix , this .suffix , Case . LOWERCASE );
277
279
}
278
280
279
281
/**
@@ -311,7 +313,7 @@ public String suffix() {
311
313
* otherwise {@code false}
312
314
*/
313
315
public boolean isUpperCase () {
314
- return Arrays . equals ( digits , UPPERCASE_DIGITS ) ;
316
+ return digitCase == Case . UPPERCASE ;
315
317
}
316
318
317
319
/**
@@ -401,16 +403,17 @@ public <A extends Appendable> A formatHex(A out, byte[] bytes, int fromIndex, in
401
403
int length = toIndex - fromIndex ;
402
404
if (length > 0 ) {
403
405
try {
404
- String between = suffix + delimiter + prefix ;
405
406
out .append (prefix );
406
407
toHexDigits (out , bytes [fromIndex ]);
407
- if (between .isEmpty ()) {
408
+ if (suffix . isEmpty () && delimiter . isEmpty () && prefix .isEmpty ()) {
408
409
for (int i = 1 ; i < length ; i ++) {
409
410
toHexDigits (out , bytes [fromIndex + i ]);
410
411
}
411
412
} else {
412
413
for (int i = 1 ; i < length ; i ++) {
413
- out .append (between );
414
+ out .append (suffix );
415
+ out .append (delimiter );
416
+ out .append (prefix );
414
417
toHexDigits (out , bytes [fromIndex + i ]);
415
418
}
416
419
}
@@ -631,7 +634,14 @@ private static String escapeNL(String string) {
631
634
* @return the hexadecimal character for the low 4 bits {@code 0-3} of the value
632
635
*/
633
636
public char toLowHexDigit (int value ) {
634
- return (char )digits [value & 0xf ];
637
+ value = value & 0xf ;
638
+ if (value < 10 ) {
639
+ return (char )('0' + value );
640
+ }
641
+ if (digitCase == Case .LOWERCASE ) {
642
+ return (char )('a' - 10 + value );
643
+ }
644
+ return (char )('A' - 10 + value );
635
645
}
636
646
637
647
/**
@@ -645,7 +655,14 @@ public char toLowHexDigit(int value) {
645
655
* @return the hexadecimal character for the bits {@code 4-7} of the value
646
656
*/
647
657
public char toHighHexDigit (int value ) {
648
- return (char )digits [(value >> 4 ) & 0xf ];
658
+ value = (value >> 4 ) & 0xf ;
659
+ if (value < 10 ) {
660
+ return (char )('0' + value );
661
+ }
662
+ if (digitCase == Case .LOWERCASE ) {
663
+ return (char )('a' - 10 + value );
664
+ }
665
+ return (char )('A' - 10 + value );
649
666
}
650
667
651
668
/**
@@ -1052,7 +1069,7 @@ public boolean equals(Object o) {
1052
1069
if (o == null || getClass () != o .getClass ())
1053
1070
return false ;
1054
1071
HexFormat otherHex = (HexFormat ) o ;
1055
- return Arrays . equals ( digits , otherHex .digits ) &&
1072
+ return digitCase == otherHex .digitCase &&
1056
1073
delimiter .equals (otherHex .delimiter ) &&
1057
1074
prefix .equals (otherHex .prefix ) &&
1058
1075
suffix .equals (otherHex .suffix );
@@ -1066,7 +1083,7 @@ public boolean equals(Object o) {
1066
1083
@ Override
1067
1084
public int hashCode () {
1068
1085
int result = Objects .hash (delimiter , prefix , suffix );
1069
- result = 31 * result + Boolean .hashCode (Arrays . equals ( digits , UPPERCASE_DIGITS ) );
1086
+ result = 31 * result + Boolean .hashCode (digitCase == Case . UPPERCASE );
1070
1087
return result ;
1071
1088
}
1072
1089
@@ -1078,7 +1095,7 @@ public int hashCode() {
1078
1095
*/
1079
1096
@ Override
1080
1097
public String toString () {
1081
- return escapeNL ("uppercase: " + Arrays . equals ( digits , UPPERCASE_DIGITS ) +
1098
+ return escapeNL ("uppercase: " + ( digitCase == Case . UPPERCASE ) +
1082
1099
", delimiter: \" " + delimiter +
1083
1100
"\" , prefix: \" " + prefix +
1084
1101
"\" , suffix: \" " + suffix + "\" " );
0 commit comments