1

我正在从 oracle db 中获取日期时间并使用 ZonedDateTime 在 java-11 中进行解析,如下所示:

  1. 甲骨文 -->1/19/2020 06:09:46.038631 PM

    java ZonedDateTime 0/p -->2020-01-19T18:09:46.038631Z[UTC]

  2. 甲骨文 -->1/19/2011 4:00:00.000000 AM

    java ZonedDateTime o/p --> 2011-01-19T04:00Z[UTC](因此,这里的 0 默认被截断。但是,我的要求是像 #1 一样具有一致的固定长度 o/p。)

预期的 java ZonedDateTime o/p -->2011-01-19T04:00:00.000000Z[UTC]

但是,我没有找到任何日期 api 方法来实现上述预期的 o/p。有没有办法以固定长度保留尾随 0,而不是操纵字符串。我们在应用程序中有一致的 ZonedDateTime 类型,因此我们不希望更改它。

4

1 回答 1

2

我们在应用程序中有一致的 ZonedDateTime 类型,因此我们不希望更改它。

为什么你认为2011-01-19T04:00Z[UTC]不一致?日期时间对象应该只保存(并提供操作的方法/函数)日期、时间和时区信息。它不应该存储任何格式信息;否则,将违反单一职责原则。格式化应由格式化类处理,例如 DateTimeFormatter(用于现代日期时间 API)、DateFormat(用于旧java.util日期时间 API)等。

每个类都应该重写该toString()函数;否则,Object#toString将在打印其对象时返回。AZonedDateTime具有日期、时间和时区信息。下面给出了它toString()的时间部分是如何实现的:

@Override
public String toString() {
    StringBuilder buf = new StringBuilder(18);
    int hourValue = hour;
    int minuteValue = minute;
    int secondValue = second;
    int nanoValue = nano;
    buf.append(hourValue < 10 ? "0" : "").append(hourValue)
        .append(minuteValue < 10 ? ":0" : ":").append(minuteValue);
    if (secondValue > 0 || nanoValue > 0) {
        buf.append(secondValue < 10 ? ":0" : ":").append(secondValue);
        if (nanoValue > 0) {
            buf.append('.');
            if (nanoValue % 1000_000 == 0) {
                buf.append(Integer.toString((nanoValue / 1000_000) + 1000).substring(1));
            } else if (nanoValue % 1000 == 0) {
                buf.append(Integer.toString((nanoValue / 1000) + 1000_000).substring(1));
            } else {
                buf.append(Integer.toString((nanoValue) + 1000_000_000).substring(1));
            }
        }
    }
    return buf.toString();
}

如您所见,第二个和 nano 部分仅在大于 时才包含在返回的字符串中0。这意味着如果要在输出字符串中使用这些(第二个和纳米)零,则需要使用格式化类。下面给出一个例子:

    import java.time.LocalDateTime;
    import java.time.ZoneOffset;
    import java.time.ZonedDateTime;
    import java.time.format.DateTimeFormatter;
    import java.time.format.DateTimeFormatterBuilder;
    import java.util.Locale;

    public class Main {
        public static void main(String[] args) {
            String input = "1/19/2011 4:00:00.000000 AM";

            // Formatter for input string
            DateTimeFormatter inputFormatter = new DateTimeFormatterBuilder()
                                                .parseCaseInsensitive()
                                                .appendPattern("M/d/u H:m:s.n a")
                                                .toFormatter(Locale.ENGLISH);

            ZonedDateTime zdt = LocalDateTime.parse(input, inputFormatter).atZone(ZoneOffset.UTC);

            // Print `zdt` in default format i.e. the string returned by `zdt.toString()`
            System.out.println(zdt);

            // Formatter for input string
            DateTimeFormatter outputFormatter = DateTimeFormatter.ofPattern("uuuu-MM-dd'T'HH:mm:ss.nnnnnnz");
            String output = zdt.format(outputFormatter);
            System.out.println(output);
        }
    }

输出:

2011-01-19T04:00Z
2011-01-19T04:00:00.000000Z

深思熟虑:

public class Main {
    public static void main(String[] args) {
        double d = 5.0000;
        System.out.println(d);
    }
}

您期望从上面给出的代码中得到什么输出?是否5.0代表不同于 的值5.0000?你将如何打印5.0000?[提示:检查String#format,NumberFormatBigDecimal]

于 2020-09-17T10:35:39.607 回答