8317742: ISO Standard Date Format implementation consistency on DateTimeFormatter and String.format

Reviewed-by: rriggs, naoto
This commit is contained in:
Shaojin Wen 2023-11-21 17:00:18 +00:00 committed by Naoto Sato
parent c4aba87570
commit 61d81d6496
2 changed files with 83 additions and 2 deletions

View File

@ -56,6 +56,7 @@ import java.time.DateTimeException;
import java.time.Instant; import java.time.Instant;
import java.time.ZoneId; import java.time.ZoneId;
import java.time.ZoneOffset; import java.time.ZoneOffset;
import java.time.chrono.IsoChronology;
import java.time.temporal.ChronoField; import java.time.temporal.ChronoField;
import java.time.temporal.TemporalAccessor; import java.time.temporal.TemporalAccessor;
import java.time.temporal.TemporalQueries; import java.time.temporal.TemporalQueries;
@ -2050,6 +2051,11 @@ public final class Formatter implements Closeable, Flushable {
return locale == null ? ',' : getDecimalFormatSymbols(locale).getGroupingSeparator(); return locale == null ? ',' : getDecimalFormatSymbols(locale).getGroupingSeparator();
} }
// Use minus sign from cached DecimalFormatSymbols.
private static char getMinusSign(Locale locale) {
return locale == null ? '-' : getDecimalFormatSymbols(locale).getMinusSign();
}
private Appendable a; private Appendable a;
private final Locale l; private final Locale l;
private IOException lastException; private IOException lastException;
@ -4490,7 +4496,20 @@ public final class Formatter implements Closeable, Flushable {
} }
case DateTime.ISO_STANDARD_DATE: { // 'F' (%Y-%m-%d) case DateTime.ISO_STANDARD_DATE: { // 'F' (%Y-%m-%d)
char sep = '-'; char sep = '-';
print(fmt, sb, t, DateTime.YEAR_4, l).append(sep); ChronoField yearField;
if (t.query(TemporalQueries.chronology()) instanceof IsoChronology) {
yearField = ChronoField.YEAR;
} else {
yearField = ChronoField.YEAR_OF_ERA;
}
int year = t.get(yearField);
if (year < 0) {
sb.append(getMinusSign(l));
year = -year;
} else if (year > 9999) {
sb.append('+');
}
sb.append(localizedMagnitude(fmt, null, year, Flags.ZERO_PAD, 4, l)).append(sep);
print(fmt, sb, t, DateTime.MONTH, l).append(sep); print(fmt, sb, t, DateTime.MONTH, l).append(sep);
print(fmt, sb, t, DateTime.DAY_OF_MONTH_0, l); print(fmt, sb, t, DateTime.DAY_OF_MONTH_0, l);
break; break;

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2003, 2022, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2003, 2023, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -34,6 +34,14 @@ import java.io.*;
import java.math.BigDecimal; import java.math.BigDecimal;
import java.math.BigInteger; import java.math.BigInteger;
import java.text.DateFormatSymbols; import java.text.DateFormatSymbols;
import java.text.DecimalFormatSymbols;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.OffsetDateTime;
import java.time.ZonedDateTime;
import java.time.ZoneOffset;
import java.time.chrono.*;
import java.time.temporal.ChronoField;
import java.util.*; import java.util.*;
import static java.util.Calendar.*; import static java.util.Calendar.*;
@ -450,5 +458,59 @@ public class BasicDateTime extends Basic {
tryCatch("%%%", UnknownFormatConversionException.class); tryCatch("%%%", UnknownFormatConversionException.class);
// perhaps an IllegalFormatArgumentIndexException should be defined? // perhaps an IllegalFormatArgumentIndexException should be defined?
tryCatch("%<%", IllegalFormatFlagsException.class); tryCatch("%<%", IllegalFormatFlagsException.class);
// %tF LocalDate
test("%tF", "2023-01-13", LocalDate.of(2023, 1, 13));
test("%tF", "2023-10-03", LocalDate.of(2023, 10, 3));
test("%tF", "0001-10-03", LocalDate.of(1, 10, 3));
test("%tF", "0012-10-03", LocalDate.of(12, 10, 3));
test("%tF", "0123-10-03", LocalDate.of(123, 10, 3));
test("%tF", "+12345-10-03", LocalDate.of(12345, 10, 3));
test("%tF", "+12345-10-03", LocalDateTime.of(12345, 10, 3, 0, 0, 0));
test("%tF", "+12345-10-03", OffsetDateTime.of(LocalDateTime.of(12345, 10, 3, 0, 0, 0), ZoneOffset.UTC));
test("%tF", "+12345-10-03", ZonedDateTime.of(LocalDateTime.of(12345, 10, 3, 0, 0, 0), ZoneOffset.UTC));
test("%tF", "-0001-10-03", LocalDate.of(-1, 10, 3));
test("%tF", "-0012-10-03", LocalDate.of(-12, 10, 3));
test("%tF", "-0123-10-03", LocalDate.of(-123, 10, 3));
test("%tF", "-1234-10-03", LocalDate.of(-1234, 10, 3));
test("%tF", "-12345-10-03", LocalDate.of(-12345, 10, 3));
test("%tF", "-12345-10-03", LocalDate.of(-12345, 10, 3));
test("%tF", "-12345-10-03", LocalDateTime.of(-12345, 10, 3, 0, 0, 0));
test("%tF", "-12345-10-03", OffsetDateTime.of(LocalDateTime.of(-12345, 10, 3, 0, 0, 0), ZoneOffset.UTC));
test("%tF", "-12345-10-03", ZonedDateTime.of(LocalDateTime.of(-12345, 10, 3, 0, 0, 0), ZoneOffset.UTC));
// check minusSign
int year = 2023, month = 1, dayOfMonth = 13;
String specifier = "%tF";
for (Locale locale : Locale.getAvailableLocales()) {
char minusSign = DecimalFormatSymbols.getInstance(locale).getMinusSign();
String str = new Formatter(new StringBuilder(), locale)
.format(specifier, LocalDate.of(year, month, dayOfMonth))
.toString();
test(locale, specifier, minusSign + str, LocalDate.of(-year, month, dayOfMonth));
}
// ja-JP-u-ca-japanese
ChronoLocalDate jpDate = Chronology
.ofLocale(Locale.forLanguageTag("ja-JP-u-ca-japanese"))
.dateNow();
test(Locale.JAPANESE,
"%tF",
String.format(
"%04d-%02d-%02d",
jpDate.get(ChronoField.YEAR_OF_ERA),
jpDate.get(ChronoField.MONTH_OF_YEAR),
jpDate.get(ChronoField.DAY_OF_MONTH)),
jpDate);
ChronoLocalDate jpDate1 = JapaneseChronology.INSTANCE.dateNow();
test(Locale.JAPANESE,
"%tF",
String.format(
"%04d-%02d-%02d",
jpDate1.get(ChronoField.YEAR_OF_ERA),
jpDate1.get(ChronoField.MONTH_OF_YEAR),
jpDate1.get(ChronoField.DAY_OF_MONTH)),
jpDate1);
} }
} }