深入了解Java8新特性-日期时间API之ZonedDateTime类

news2025/1/20 13:23:04

阅读建议

嗨,伙计!刷到这篇文章咱们就是有缘人,在阅读这篇文章前我有一些建议:

  1. 本篇文章大概19000多字,预计阅读时间长需要10分钟以上。
  2. 本篇文章的实战性、理论性较强,是一篇质量分数较高的技术干货文章,建议收藏起来,方便时常学习与回顾,温故而知新。
  3. 创作不易,免费的点赞、关注,请走上一走,算是对博主一些鼓励,让我更有动力输出更多的干货内容。

注意

本文编写的单元测试是基于java11,具体的版本号是:11.0.19

关于ZonedDateTime

ZonedDateTime是Java 8中引入的一个不可变且线程安全的数据类型,用于表示带有时区的日期和时间。与LocalDateTime相比,ZonedDateTime可以存储时区信息,并能够处理模糊的本地日期时间。

ZonedDateTime可以表示从本地时间线LocalDateTime到即时时间线Instant之间的日期和时间。两个时间线之间的差异是UTC /格林威治的偏移量,由ZoneOffset表示。在两个时间线之间转换涉及使用从ZoneId访问的ZoneId计算偏移量。

ZonedDateTime的常用方法包括获取年份、月份、日期、小时、分钟、秒等信息的get方法,以及计算两个日期之间的差值的minus方法等。ZonedDateTime还提供了用于获取当前日期和时间的now方法,以及用于将ZonedDateTime转换为其它日期/时间类型的toLocalDate、toLocalTime等方法。

需要注意的是,ZonedDateTime在进行时区转换时可能会遇到夏令时调整的情况,此时需要使用正确的时区信息以避免错误的出现。另外,在处理涉及时区的日期和时间时,需要考虑时区偏移量以及时间线之间的差异等问题。

核心方法

ZonedDateTime#now()

ZonedDateTime#now() 用于获取当前时间。它返回一个表示当前日期和时间的ZonedDateTime 对象,包括时区和偏移量。

@Test
public void test() {
    ZonedDateTime zonedDateTime = ZonedDateTime.now();
    System.out.println(zonedDateTime);//输出结果:2023-11-29T13:31:38.124890400+08:00[Asia/Shanghai]
}

ZonedDateTime#of(...)、ZonedDateTime#ofLocal(...)、ZonedDateTime#ofInstant(...)、ZonedDateTime#ofStrict(...)

  • ZonedDateTime#of(...): 该方法用于从年、月、日、时、分、秒和时区中获取ZonedDateTime的实例。所有这些值分别是年、月、日、时、分、秒和时区作为参数传递。
  • ZonedDateTime#ofLocal(LocalDateTime, ZoneId, ZoneOffset): 该方法用于使用ZonmeId和首选偏移量从本地日期时间创建ZonedDateTime实例。
  • ZonedDateTime#ofInstant(Instant, ZoneId): 该方法用于从Instant和ZoneId中获取ZonedDateTime实例。
  • ZonedDateTime#ofStrict(LocalDateTime, ZoneId, ZoneOffset): 该方法用于从严格模式下的本地日期时间创建ZonedDateTime实例。
@Test
public void test2() {
    ZoneId zoneId = ZoneId.systemDefault();
    System.out.println(zoneId.getId());//输出结果:Asia/Shanghai
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    System.out.println(zonedDateTime);//输出结果:2023-11-29T13:32:12+08:00[Asia/Shanghai]
    LocalDate localDate = LocalDate.of(2023, 11, 29);
    LocalTime localTime = LocalTime.of(13, 32, 12, 0);
    ZonedDateTime zonedDateTime1 = ZonedDateTime.of(localDate, localTime, zoneId);
    System.out.println(zonedDateTime1);//输出结果:2023-11-29T13:32:12+08:00[Asia/Shanghai]
}

ZonedDateTime#from(...)

ZonedDateTime#from() 用于从其他时间单位(如 LocalDateTime, LocalDate, LocalTime, Instant 等)创建 ZonedDateTime 对象。这个方法接收一个 TemporalAccessor 对象,然后根据该对象的时间单位,将其转换为 ZonedDateTime 对象。例如,如果我们有一个 OffsetDateTime 对象,我们可以使用 ZonedDateTime#from() 方法将其转换为 ZonedDateTime 对象:

@Test
public void test3() {
    OffsetDateTime offsetDateTime = OffsetDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneOffset.ofHours(-5));
    ZonedDateTime from = ZonedDateTime.from(offsetDateTime);
    System.out.println(from);//输出结果:2023-11-29T13:32:12-05:00
}

ZonedDateTime#parse(...)

ZonedDateTime#parse() 用于将字符串解析为 ZonedDateTime 对象。这个方法接受一个字符串参数,并尝试将其解析为 

ZonedDateTime 格式。这里要特别注意:如果字符格式不符合规范,则会抛出异常;

@Test
public void test4() {
    String str = "2023-11-29T13:32:12+08:00[Asia/Shanghai]";
    ZonedDateTime zonedDateTime = ZonedDateTime.parse(str);
    System.out.println(zonedDateTime);//输出结果:2023-11-29T13:32:12+08:00[Asia/Shanghai]
}

ZonedDateTime#isSupported(...)

ZonedDateTime#isSupported(...) 用于确定当前时间是否支持给定的时间字段。这个方法接受一个 TemporalField 对象作为参数,如果当前时间支持该字段,则返回 true,否则返回 false。

@Test
public void test5() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    boolean supported = zonedDateTime.isSupported(ChronoUnit.YEARS);
    boolean supported2 = zonedDateTime.isSupported(ChronoField.YEAR);
    System.out.println(supported);//输出结果:true
    System.out.println(supported2);//输出结果:true
}

ZonedDateTime#range(...)

ZonedDateTime#range(...) 是 用于获取给定时间字段的范围。这个方法接受一个 TemporalField 对象作为参数,返回一个 TemporalRange 对象,表示该字段的范围。

@Test
public void test6() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ValueRange range = zonedDateTime.range(ChronoField.DAY_OF_MONTH);
    long minimum = range.getMinimum();
    long maximum = range.getMaximum();
    System.out.println(minimum);//输出结果:true
    System.out.println(maximum);//输出结果:true
}

ZonedDateTime#get(...)、ZonedDateTime#getLong(...)、ZonedDateTime#getOffset(...)

  • ZonedDateTime#get(...):该方法用于从ZonedDateTime对象获取指定字段的值。例如,调用get()方法并传入一个表示“小时”的字段,将返回当前时间的小时值。如果查询的字段值超出该字段的有效范围,或者不支持该字段,可能会引发异常。
  • ZonedDateTime#getLong(...):这个方法用于获取ZonedDateTime对象中指定字段的长整型值。同样,如果查询的字段值超出有效范围,或者不支持该字段,可能会引发异常。
  • ZonedDateTime#getOffset(...):此方法用于获取ZonedDateTime对象的时区偏移量。它返回一个表示偏移量的ZoneOffset对象。

这些方法都可用于处理和操作具有时区信息的时间,但get()和getLong()方法更侧重于获取时间的特定部分,而getOffset()则关注于获取时区的偏移量。

@Test
public void test7() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    int year = zonedDateTime.get(ChronoField.YEAR);
    long yearOfYear = zonedDateTime.getLong(ChronoField.YEAR);
    int month = zonedDateTime.get(ChronoField.MONTH_OF_YEAR);
    int day = zonedDateTime.get(ChronoField.DAY_OF_MONTH);
    int dayOfYear = zonedDateTime.get(ChronoField.DAY_OF_YEAR);
    ZoneOffset offset = zonedDateTime.getOffset();
    System.out.println(year);//输出结果:2023
    System.out.println(yearOfYear);//输出结果:2023
    System.out.println(month);//输出结果:11
    System.out.println(day);//输出结果:29
    System.out.println(dayOfYear);//输出结果:333
    System.out.println(offset);//输出结果:+08:00
}

ZonedDateTime#getZone(...)

ZonedDateTime#getZone() 用于获取该对象的时区。此方法返回一个 ZoneId 对象,表示 ZonedDateTime 对象的时区。ZoneId 是 Java 8 中表示时区的类。

    @Test
    public void test8() {
        ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("America/New_York"));
        String zoneId = zonedDateTime.getZone().getId();
        System.out.println(zoneId);//输出结果:America/New_York
    }

ZonedDateTime#withZoneSameLocal(...)、ZonedDateTime#withZoneSameInstant(...)、ZonedDateTime#withFixedOffsetZone(...)

  • ZonedDateTime#withZoneSameLocal():用于通过更改时区来返回此ZonedDateTime对象的副本,如果可能的话不更改本地日期时间。仅当本地日期时间对于新区域无效时才更改本地日期时间。此方法尽量保持相同的本地时间,同时改变时区。
  • 使用场景:当你需要确保在特定时区下的本地时间不变,而仅仅需要改变日期的表示时区时,可以使用此方法。
  • ZonedDateTime#withZoneSameInstant():这个方法通过更改时区且不带瞬时值来返回此ZonedDateTime对象的副本。换句话说,它将尝试保持相同的时间戳,但可能在不同的时区下。当你需要确保在特定时区下的时间戳不变,而仅仅需要改变日期的表示时区时,可以使用此方法。
  • ZonedDateTime#withFixedOffsetZone():这个方法允许你使用固定的偏移量来创建新的ZonedDateTime对象。偏移量表示从UTC(协调世界时间)到当前时区的偏移量。当你需要创建一个具有特定偏移量的ZonedDateTime对象,而不管具体的时区是什么时,可以使用此方法。
@Test
public void test9() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ZonedDateTime zonedDateTime1 = zonedDateTime.withZoneSameLocal(ZoneId.of(("America/New_York")));
    System.out.println(zonedDateTime1);//输出结果:2023-11-29T13:32:12-05:00[America/New_York]
    ZonedDateTime zonedDateTime2 = zonedDateTime.withZoneSameInstant(ZoneId.of(("America/New_York")));
    System.out.println(zonedDateTime2);//输出结果:2023-11-29T00:32:12-05:00[America/New_York]
    ZonedDateTime zonedDateTime3 = zonedDateTime.withFixedOffsetZone();
    System.out.println(zonedDateTime3);//输出结果:2023-11-29T13:32:12+08:00
}

ZonedDateTime#toLocalDateTime(...)、ZonedDateTime#toLocalDate(...)、ZonedDateTime#toLocalTime(...)

ZonedDateTime类的toLocalDateTime()方法、toLocalDate()方法和toLocalTime()方法分别用于返回该日期时间对象的LocalDateTime部分、LocalDate部分和LocalTime部分。以下是每个方法的功能作用和使用场景:

  • ZonedDateTime#toLocalDateTime():该方法返回一个与当前ZonedDateTime对象日期和时间部分相同的LocalDateTime对象。它主要用于需要同时获取日期和时间但不需要时区和偏移量信息的情况。例如,在记录事件发生的时间、计算时间间隔或进行日期时间运算时,可以使用该方法。
  • ZonedDateTime#toLocalDate():该方法返回一个与当前ZonedDateTime对象日期部分相同的LocalDate对象。它主要用于需要获取日期但不需要具体时间信息的情况。例如,在记录一个事件发生的日期、计算日期间隔或进行日期运算时,可以使用该方法。
  • ZonedDateTime#toLocalTime():该方法返回一个与当前ZonedDateTime对象时间部分相同的LocalTime对象。它主要用于需要获取时间但不需要具体日期信息的情况。例如,在记录一个事件发生的时间、计算时间间隔或进行时间运算时,可以使用该方法。

这些方法提供了一种灵活的方式来获取和处理日期和时间的不同组成部分。通过将ZonedDateTime对象转换为LocalDateTime、LocalDate或LocalTime对象,可以更方便地进行各种日期和时间相关的操作。

@Test
public void test10() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    LocalDateTime localDateTime = zonedDateTime.toLocalDateTime();
    System.out.println(localDateTime);//输出结果:2023-11-29T13:32:12
    LocalDate localDate = zonedDateTime.toLocalDate();
    System.out.println(localDate);//输出结果:2023-11-29
    LocalTime localTime = zonedDateTime.toLocalTime();
    System.out.println(localTime);//输出结果:13:32:12
}

ZonedDateTime#getYear(...)、ZonedDateTime#getMonthValue(...)、ZonedDateTime#getMonth(...)、ZonedDateTime#getDayOfMonth(...)、ZonedDateTime#getDayOfYear(...)、ZonedDateTime#getDayOfWeek(...)、ZonedDateTime#getHour(...)、ZonedDateTime#getMinute(...)、ZonedDateTime#getSecond(...)、ZonedDateTime#getNano(...)

  • ZonedDateTime#getYear(...):这个方法的功能是获取年份字段。它返回该年份的原始int值。此方法返回的年份是proleptic,即根据UTC/格林威治标准时间相对应的本地日期时间与区域ID规则定义的有效偏移量来获得。
  • ZonedDateTime#getMonthValue(...):这个方法的功能是获取月份字段的值。它返回一个表示月份的整数,范围从1(一月)到12(十二月)。
  • ZonedDateTime#getMonth(...):这个方法的功能是获取月份字段。它返回一个表示月份的枚举值,如JANUARY、FEBRUARY等。
  • ZonedDateTime#getDayOfMonth(...):这个方法的功能是获取一个月中的日期字段。它返回一个表示一个月中某一天的整数,范围从1到该月的最大天数。
  • ZonedDateTime#getDayOfYear(...):这个方法的功能是获取一年中的日期字段。它返回一个表示一年中某一天的整数,范围从1到365(或366在一些年份)。
  • ZonedDateTime#getDayOfWeek(...):这个方法的功能是获取一周中的日期字段。它返回一个表示星期几的枚举值,如SUNDAY、MONDAY等。
  • ZonedDateTime#getHour(...):这个方法的功能是获取小时字段。它返回一个表示小时的整数,范围从0(午夜)到23(晚上11点)。
  • ZonedDateTime#getMinute(...):这个方法的功能是获取分钟字段。它返回一个表示分钟的整数,范围从0到59。
  • ZonedDateTime#getSecond(...):这个方法的功能是获取秒字段。它返回一个表示秒数的整数,范围从0到59。
  • ZonedDateTime#getNano(...):这个方法的功能是获取毫秒字段的一部分(即纳秒)。它返回一个表示纳秒数的整数,范围从0到999999999。

这些方法的作用是根据给定的时间点获取不同的时间单位信息。主要的使用场景:

  • 获取特定时间点的年、月、日、时、分、秒和纳秒等信息。
  • 在日期和时间计算中,使用这些方法来分析和操作日期和时间数据。
  • 在需要精确到纳秒级别的计算或记录中,使用这些方法来获取和操作时间信息。
@Test
public void test11() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    int year = zonedDateTime.getYear();
    int monthValue = zonedDateTime.getMonthValue();
    int dayOfMonth = zonedDateTime.getDayOfMonth();
    int dayOfYear = zonedDateTime.getDayOfYear();
    int hour = zonedDateTime.getHour();
    System.out.println(year);//输出结果:2023
    System.out.println(monthValue);//输出结果:11
    System.out.println(dayOfMonth);//输出结果:29
    System.out.println(dayOfYear);//输出结果:333
    System.out.println(hour);//输出结果:13
}

ZonedDateTime#with(...)、ZonedDateTime#withYear(...)、ZonedDateTime#withMonth(...)、ZonedDateTime#withDayOfMonth(...)、ZonedDateTime#withDayOfYear(...)、ZonedDateTime#withHour(...)、ZonedDateTime#withMinute(...)、ZonedDateTime#withSecond(...)、ZonedDateTime#withNano(...)

  • ZonedDateTime#with(TemporalAdjuster adjuster):该方法允许您使用指定的调整器对象对日期/时间进行调整。这可以用于执行各种日期/时间相关的操作,例如更改日期、时间或时区等。返回的是一个新的ZonedDateTime对象,它是原始对象的副本,不会影响原始对象。
  • ZonedDateTime#withYear(int year):这个方法用于更改ZonedDateTime中的年份。它返回一个新的ZonedDateTime对象,其中包含更改后的年份和原始的月份、日期、小时、分钟、秒和纳秒。此方法在本地时间轴上操作,更改本地日期时间的年份以及此操作之后的本地日期时间使用区域ID获取偏移量,将其转换回ZonedDateTime。当转换回ZonedDateTime时,如果本地日期时间重叠,则将尽可能保留偏移量,否则将使用较早的偏移量。
  • ZonedDateTime#withMonth(int month):这个方法用于更改ZonedDateTime中的月份。它返回一个新的ZonedDateTime对象,其中包含更改后的月份和原始的年份、日期、小时、分钟、秒和纳秒。
  • ZonedDateTime#withDayOfMonth(int dayOfMonth):这个方法用于更改ZonedDateTime中的日期。它返回一个新的ZonedDateTime对象,其中包含更改后的日期和原始的年份、月份、小时、分钟、秒和纳秒。
  • ZonedDateTime#withDayOfYear(int dayOfYear):这个方法用于更改ZonedDateTime中的一年中的日期。它返回一个新的ZonedDateTime对象,其中包含更改后的日期和原始的年份、月份、小时、分钟、秒和纳秒。
  • ZonedDateTime#withHour(int hour):这个方法用于更改ZonedDateTime中的小时。它返回一个新的ZonedDateTime对象,其中包含更改后的小时和原始的年份、月份、日期、分钟、秒和纳秒。
  • ZonedDateTime#withMinute(int minute):这个方法用于更改ZonedDateTime中的分钟。它返回一个新的ZonedDateTime对象,其中包含更改后的分钟和原始的年份、月份、日期、小时、秒和纳秒。
  • ZonedDateTime#withSecond(int second):这个方法用于更改ZonedDateTime中的秒数。它返回一个新的ZonedDateTime对象,其中包含更改后的秒数和原始的年份、月份、日期、小时、分钟和纳秒。
  • ZonedDateTime#withNano(int nano):这个方法用于更改ZonedDateTime中的纳秒数。它返回一个新的ZonedDateTime对象,其中包含更改后的纳秒数和原始的年份、月份、日期、小时、分钟和秒。

这些方法的使用场景包括但不限于以下情况:

  • 在需要对日期/时间进行精细操作或调整时,例如更改年、月、日等字段或执行更复杂的日期/时间计算时,可以使用这些方法。
  • 在需要创建具有特定日期/时间信息的副本或新的日期/时间对象时,可以使用这些方法。
  • 在需要进行时区转换或处理时,可以使用这些方法来获取特定时区下的日期/时间信息。
@Test
public void test12() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ZonedDateTime zonedDateTime1 = zonedDateTime.with(ChronoField.YEAR, 2022);
    System.out.println(zonedDateTime1);//输出结果2022-11-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime2 = zonedDateTime.withYear(2022);
    System.out.println(zonedDateTime2);//输出结果2022-11-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime3 = zonedDateTime.with(ChronoField.DAY_OF_MONTH, 1);
    System.out.println(zonedDateTime3);//输出结果2023-11-01T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime4 = zonedDateTime.with(ChronoField.HOUR_OF_DAY, 1);
    System.out.println(zonedDateTime4);//输出结果:2023-11-29T01:32:12+08:00[Asia/Shanghai]
}

ZonedDateTime#truncatedTo(...)

ZonedDateTime#truncatedTo(TemporalUnit unit) 的主要作用是将当前的ZonedDateTime对象截断到给定的时间单位,并返回一个新的ZonedDateTime对象。具体来说,这个方法接受一个 TemporalUnit 参数,该参数表示要截断的时间单位。可能的值包括 DAYS、HOURS、MICROSECONDS、MILLISECONDS、MINUTES、NANOSECONDS、SECONDS 等。这个方法将忽略时间单位比指定单位更小的时间单位,例如,如果将日期时间截断到分钟,那么该方法将忽略秒和毫秒字段。

@Test
public void test13() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ZonedDateTime zonedDateTime1 = zonedDateTime.truncatedTo(ChronoUnit.DAYS);
    System.out.println(zonedDateTime1);//输出结果:2023-11-29T00:00+08:00[Asia/Shanghai]
}

ZonedDateTime#plus(...)、ZonedDateTime#plusYears(...)、ZonedDateTime#plusMonths(...)、ZonedDateTime#plusWeeks(...)、ZonedDateTime#plusDays(...)、ZonedDateTime#plusHours(...)、ZonedDateTime#plusMinutes(...)、ZonedDateTime#plusSeconds(...)、ZonedDateTime#plusNanos(...)

  • ZonedDateTime#plus(...):此方法用于返回此日期时间的副本,其中添加了指定的单位数量。它接受两个参数:amountToAdd(要添加到结果中的单位数量,可能为负)和unit(要添加数量的单位)。返回值是根据此日期时间返回的ZonedDateTime,并添加了指定的数量。该数量通常为Period或Duration,但可以是实现TemporalAmount接口的任何其他类型。当由于不支持该单位或由于其他原因而无法添加数量时,会抛出异常。
  • ZonedDateTime#plusYears(...):此方法用于在此ZonedDateTime中添加年数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusMonths(...):此方法用于在此ZonedDateTime中添加月数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusWeeks(...):此方法用于在此ZonedDateTime中添加周数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusDays(...):此方法用于在此ZonedDateTime中添加天数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusHours(...):此方法用于在此ZonedDateTime中添加小时数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusMinutes(...):此方法用于在此ZonedDateTime中添加分钟数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusSeconds(...):此方法用于在此ZonedDateTime中添加秒数,并在添加后返回ZonedDateTime的副本。
  • ZonedDateTime#plusNanos(...):此方法用于在此ZonedDateTime中添加纳秒数,并在添加后返回ZonedDateTime的副本。
@Test
public void test14() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ZonedDateTime zonedDateTime1 = zonedDateTime.plus(1, ChronoUnit.YEARS);
    System.out.println(zonedDateTime1);//输出结果:2024-11-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime2 = zonedDateTime.plusYears(1);
    System.out.println(zonedDateTime2);//输出结果:2024-11-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime3 = zonedDateTime.plusMonths(1);
    System.out.println(zonedDateTime3);//输出结果:2023-12-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime4 = zonedDateTime.plusDays(1);
    System.out.println(zonedDateTime4);//输出结果:2023-11-30T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime5 = zonedDateTime.plusMinutes(28);
    System.out.println(zonedDateTime5); //输出结果:2023-11-29T14:00:12+08:00[Asia/Shanghai]
}

ZonedDateTime#minus(...)、ZonedDateTime#minusYears(...)、ZonedDateTime#minusMonths(...)、ZonedDateTime#minusWeeks(...)、ZonedDateTime#minusDays(...)、ZonedDateTime#minusHours(...)、ZonedDateTime#minusMinutes(...)、ZonedDateTime#minusSeconds(...)、ZonedDateTime#minusNanos(...)

  • ZonedDateTime#minus(...):此方法用于返回此日期时间的副本,其中减去指定的单位数量。它接受两个参数:amountToSubtract表示要减去的单位量,可能为负数;unit表示要减去的量的单位。此方法基于本地时间轴进行操作,将本地日期时间减去指定的年、月、日、时、分、秒和纳秒等单位。使用此方法时,需要确保减去的结果在支持的日期范围内,否则可能会抛出异常。
  • ZonedDateTime#minusYears(...):此方法用于从当前日期时间减去指定数量的年份。它接受一个表示年份的参数,该参数可以是负数。此方法在本地时间轴上操作,将从本地日期时间中减去年份,并使用区域ID获取偏移量。当转换回ZonedDateTime时,如果本地日期时间重叠,将尽可能保留偏移量,否则将使用较早的偏移量。此实例是不可变的,不受此方法调用的影响。
  • ZonedDateTime#minusMonths(...):此方法用于从当前日期时间减去指定数量的月份。它接受一个表示月份的参数。此方法在本地时间轴上操作,将从本地日期时间中减去月份。
  • ZonedDateTime#minusWeeks(...):此方法用于从当前日期时间减去指定数量的周数。它接受一个表示周数的参数。此方法在本地时间轴上操作,将从本地日期时间中减去周数。
  • ZonedDateTime#minusDays(...):此方法用于从当前日期时间减去指定数量的天数。它接受一个表示天数的参数。此方法在本地时间轴上操作,将从本地日期时间中减去天数。
  • ZonedDateTime#minusHours(...):此方法用于从当前日期时间减去指定的小时数。它接受一个表示小时数的参数。此方法在本地时间轴上操作,将从本地日期时间中减去小时数。
  • ZonedDateTime#minusMinutes(...):此方法用于从当前日期时间减去指定的分钟数。它接受一个表示分钟数的参数。此方法在本地时间轴上操作,将从本地日期时间中减去分钟数。
  • ZonedDateTime#minusSeconds(...):此方法用于从当前日期时间减去指定的秒数。它接受一个表示秒数的参数。此方法在本地时间轴上操作,将从本地日期时间中减去秒数。
  • ZonedDateTime#minusNanos(...):此方法用于从当前日期时间减去指定的纳秒数。它接受一个表示纳秒数的参数。此方法在本地时间轴上操作,将从本地日期时间中减去纳秒数。
@Test
public void test15() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ZonedDateTime zonedDateTime1 = zonedDateTime.minus(1, ChronoUnit.YEARS);
    System.out.println(zonedDateTime1);//输出结果:2022-11-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime2 = zonedDateTime.minusYears(1);
    System.out.println(zonedDateTime2);//输出结果:2022-11-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime3 = zonedDateTime.minusMonths(1);
    System.out.println(zonedDateTime3);//输出结果:2023-10-29T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime4 = zonedDateTime.minusDays(1);
    System.out.println(zonedDateTime4);//输出结果:2023-11-28T13:32:12+08:00[Asia/Shanghai]
    ZonedDateTime zonedDateTime5 = zonedDateTime.minusMinutes(28);
    System.out.println(zonedDateTime5); //输出结果:2023-11-29T13:04:12+08:00[Asia/Shanghai]
}

ZonedDateTime#query(...)

ZonedDateTime#query(...) 用于查询与当前 ZonedDateTime 对象相关的信息。该方法接受一个 TemporalQuery 参数,用于指定要查询的信息类型。TemporalQuery 是一个函数式接口,用于从给定的时间对象中提取所需的信息。

@Test
public void test16() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    TemporalQuery<Integer> temporalQuery=item->item.get(ChronoField.DAY_OF_YEAR);
    Integer dayOfYear = zonedDateTime.query(temporalQuery);
    System.out.println(dayOfYear); //输出结果:333
}

ZonedDateTime#until(...)

ZonedDateTime#until(ZonedDateTime) 用于根据单个TemporalUnit计算两个日期时间之间的时间差。

@Test
public void test17() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2022, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    ZonedDateTime zonedDateTime2 = ZonedDateTime.of(2023, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    long until = zonedDateTime.until(zonedDateTime2, ChronoUnit.MONTHS);
    System.out.println(until); //输出结果:12
}

ZonedDateTime#format(...)

ZonedDateTime#format() 用于将当前对象的时间信息按照指定的格式进行格式化,并返回一个字符串表示。该方法接受一个DateTimeFormatter 参数,用于指定日期时间的格式。

@Test
public void test18() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2022, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    String format = zonedDateTime.format(DateTimeFormatter.ISO_DATE);
    System.out.println(format); //输出结果:2022-11-29+08:00
}

ZonedDateTime#toOffsetDateTime(...)

ZonedDateTime#toOffsetDateTime() 用于将当前对象转换为一个 OffsetDateTime 对象。OffsetDateTime 是 Java 中的一个日期时间类,它表示一个带有时间偏移量的日期时间,通常用于处理具有不同时区的日期时间。

ZonedDateTime 对象表示一个具有时区的日期时间,而 OffsetDateTime 对象则表示一个具有时间偏移量的日期时间。因此,通过调用 toOffsetDateTime() 方法,可以将 ZonedDateTime 对象转换为一个 OffsetDateTime 对象,以便进行进一步的处理或操作。

@Test
public void test19() {
    ZonedDateTime zonedDateTime = ZonedDateTime.of(2022, 11, 29, 13, 32, 12, 0, ZoneId.of("Asia/Shanghai"));
    OffsetDateTime offsetDateTime = zonedDateTime.toOffsetDateTime();
    System.out.println(offsetDateTime); //输出结果:2022-11-29T13:32:12+08:00
}

使用场景

ZonedDateTime常用于表示带有时区的日期和时间,它的使用场景包括但不限于:

  • 处理具有不同时区的日期时间:由于地球是圆形的,不同地理位置处于不同的时区,因此需要在处理日期时间时考虑时区差异。ZonedDateTime可以用于表示带有时区的日期和时间,并能够根据时区进行转换和处理。
  • 日程安排和时间规划:在日程安排和时间规划中,需要考虑不同时区的时间差异。ZonedDateTime可以用于表示会议、活动等的时间和地点,并能够根据参与者所在的时区进行自动转换。
  • 全球化应用:随着全球化的发展,跨时区的交流和合作变得越来越普遍。ZonedDateTime可以用于表示全球范围内的日期和时间,并能够根据不同时区进行格式化和显示。
  • 系统集成和数据交换:在企业级应用和系统集成中,不同系统之间可能存在时区差异。ZonedDateTime可以用于表示系统之间的日期和时间,并能够根据需要进行转换和处理。

总结

与OffsetDateTime的区别

ZonedDateTime与OffsetDateTime的区别主要包括以下几个方面:

  • 时区信息:ZonedDateTime包含时区信息,可能包含城市或地区的信息,而OffsetDateTime只包含时差信息,不包含城市或地区的信息。
  • 创建方式:创建ZonedDateTime和OffsetDateTime类的对象通常通过now()和of()这两个方法完成。通过now()方法所创建的对象表示系统当前的日期时间。而如果通过of()方法创建对象,需要给of()方法传递3部分信息,分别是:日期、时间以及时区。在表示日期时间的参数方面,ZonedDateTime的of()方法的表示时区的参数是ZoneId类型,因此用ZoneRegion类对象和ZoneOffset类对象都可以充当这个参数,而OffsetDateTime的of()方法表示时区的参数类型是ZoneOffset,因此仅能用该类型的对象充当参数。
  • 存储信息:ZonedDateTime类对象中包含localdatetime、zoneid和已解析的zoneoffset三个独立对象的状态,而OffsetDateTime类对象只包含日期和时间字段,以及与GMT/UTC的偏移量。

综上所述,ZonedDateTime和OffsetDateTime都是Java 8中表示日期时间的类,它们在处理日期和时间方面有所不同。ZonedDateTime更适用于处理具有时区信息的日期和时间,而OffsetDateTime则更适用于处理具有偏移量信息的日期和时间。

与LocalDateTime的区别

ZonedDateTime与LocalDateTime的主要区别体现在以下方面:

  • 时区信息:ZonedDateTime包含时区信息,表示一个具有特定时区的日期和时间,而LocalDateTime则没有时区信息,它表示的是一个相对独立的日期和时间。
  • 创建方式:创建ZonedDateTime和LocalDateTime类的对象通常是通过now()和of()这两个方法完成的。通过now()方法所创建的对象表示系统当前的日期时间。而如果通过of()方法创建对象,需要给of()方法传递3部分信息,分别是:日期、时间以及时区。
  • 应用场景:ZonedDateTime更适用于处理具有时区信息的日期和时间,而LocalDateTime则更适用于处理具有偏移量信息的日期和时间。
  • 不可变性:ZonedDateTime是不可变的,即一旦创建了ZonedDateTime对象,其日期和时间信息就不能再被修改。这种不可变性使得ZonedDateTime对象是线程安全的。而LocalDateTime也是不可变的,它同样具有线程安全的特点。

总的来说,ZonedDateTime和LocalDateTime类都是Java 8中表示日期和时间的工具,它们都不可变且线程安全。然而,ZonedDateTime更注重时区信息的处理,而LocalDateTime则不具备这种能力。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1269079.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

群晖安装portainer

一、下载镜像 打开【Container Manager】 ,搜索portainer&#xff0c;双击【6053537/portainer-ce】下载汉化版本 二、创建映射文件夹 打开【File Station】&#xff0c;在docker目录下创建【portainer】文件夹 三、开启SSH 群晖 - 【控制面板】-【终端机和SNMP】 勾选【启动…

oracle数据库节点一宕机重启后集群crsd服务没有起

13:18:55时节点一服务器宕机后&#xff0c;节点2心跳不通剔除了节点1 之后节点1服务器重启 集群设置自动拉起&#xff0c;但节点一启动后集群在crsd服务上迟迟没有起来 去查看了crsd的日志发现这一时间点心跳一直不通 在节点一起crsd服务&#xff0c;执行以下命令 #&#x…

Cytoscape学习教程

写在前面 今天分享的内容是自己遇到问题后,咨询社群里面的同学,帮忙解决的总结。 关于Cytoscape,对于做组学或生物信息学的同学基本是陌生的,可能有的同学用这个软件作图是非常溜的,做出来的网络图也是十分的好看,“可玩性”很高,就像前面分享的aPEAR包一样aPEAR包绘制…

0基础能不能转行做网络安全?网络安全人才发展路线

最近有同学在后台留言&#xff0c;0基础怎么学网络安全&#xff1f;0基础可以转行做网络安全吗&#xff1f;以前也碰到过类似的问题&#xff0c;想了想&#xff0c;今天简单写一下。 我的回答是先了解&#xff0c;再入行。 具体怎么做呢&#xff1f; 首先&#xff0c;你要确…

羊大师不说,你会知道酸奶也能加热吗?

羊大师不说&#xff0c;你会知道酸奶也能加热吗&#xff1f; 酸奶是一种非常受欢迎的健康食品&#xff0c;不仅可以冷饮&#xff0c;还可以热着喝哦&#xff01;你可能会好奇&#xff0c;酸奶热着喝会不会破坏其营养价值呢&#xff1f;别着急&#xff0c;让小编羊大师来为你解…

基于springboot的电影院管理系统的设计与实现 (含论文和源码视频导入教程)

&#x1f449;文末查看项目功能视频演示获取源码sql脚本视频导入教程视频 1 、功能描述 基于springboot的电影院管理系统7拥有两种角色 管理员&#xff1a;用户管理、购票统计、电影管理、电影类型管理、放映厅管理、订单管理等 用户&#xff1a;登录注册、查看各种信息、购票…

Spark local模式的安装部署

安装与配置Spark开发环境。 相关知识 Apache Spark是专为大规模数据处理而设计的快速通用的计算引擎。Spark是UC Berkeley AMP lab(加州大学伯克利分校的AMP实验室)所开源的类Hadoop MapReduce的通用并行框架&#xff0c;Spark拥有Hadoop MapReduce所具有的优点&#xff1b;但…

iOS上传ipa使用可视化工具Transporter

文章目录 前言一、Transporter二、Appuploader三、iTMSTransporter总结 前言 最近为了让非开发人员上传IPA文件&#xff0c;特意找了一些方法&#xff0c;至于以前的ApplicationUploader已经不能用了&#xff0c;下面介绍两个工具可以上传IPA包。 一、Transporter 1、操作简单…

QML Column Row 属性 pyside6

在 QML 中&#xff0c;Column 和 Row 是常用的布局元素&#xff0c;用于水平&#xff08;Row&#xff09;和垂直&#xff08;Column&#xff09;排列它们的子元素。以下是这两个元素的主要属性列表&#xff1a; Column 属性 spacing: 子元素之间的垂直间隔。width 和 height:…

Image Super-Resolution with Text Prompt Diffusion

Image Super-Resolution with Text Prompt Diffusion (Paper reading) Zheng Chen, Shanghai Jiao Tong University, arXiv23, Code, Paper 1. 前言 受多模态方法和文本提示图像处理进步的启发&#xff0c;我们将文本提示引入图像SR&#xff0c;以提供退化先验。具体来说&am…

Redis 发布订阅机制深入探索

Redis 的发布订阅&#xff08;pub/sub&#xff09;机制是一种消息传递模式&#xff0c;允许消息的发送者&#xff08;发布者&#xff09;和消息的接收者&#xff08;订阅者&#xff09;通过一个中介层&#xff08;频道&#xff09;进行通信&#xff0c;而无需彼此直接交互。以下…

ARM麒麟V10 auditctl启动失败处理

问题&#xff1a; 业务服务器需要启用审计服务&#xff0c;但是启动审计服务失败&#xff0c;查看状态提示audit0。 修改配置文件/boot/efi/EFI/kylin/grub.cfg 删除audit0&#xff0c;或者设置audit1。 重启服务器后验证状态。 auditctl -D echo "-w /data -p rwxa"…

磁环电感参数计算

磁环电感参数计算 1.什么是电感磁饱和2.电感饱和的原因3.电感饱和带来的影响3.1 感应电动势变化3.2 电感值变化3.3 功率损耗增加3.4 系统稳定性受到影响4.饱和电流计算最近在做DC/DC电源,电感是用磁环绕制的,所以关注一下磁环绕制电感参数的计算,学习学习。 某款磁环参数。 …

【10】Python函数专题(上)

目录 1.什么是函数2.函数的参数2.1形参 和 实参2.2函数的传递方式2.3 不定长参数2.3.1 可变位置参数`*args`2.3.2可变关键字参数`**kwargs`2.3.3 小结2.4 参数的解包(也称拆包)1.什么是函数 在Python中,函数是一段可重复使用的代码块,用于执行特定任务。通过将代码封装在函…

C++设计模式——Bridge模式(下)

在上篇 《C设计模式——Bridge模式&#xff08;上&#xff09;》中我们对于桥接模式做了一些介绍。介于桥接模式在实际项目开发中使用广泛&#xff0c;而且也是面试中常问常新的话题。在本篇&#xff0c;我们专注bridge模式在具体的项目开发中的应用&#xff0c;举几个例子来说…

Linux系统安装-以文本模式安装rhel8

文本模式安装提供了用于安装 Red Hat Enterprise Linux 的交互式非图形界面。此安装方法对于没有图形功能的系统很有用。但是&#xff0c;在开始基于文本的安装之前&#xff0c;请务必考虑可用的替代方案。文本模式在安装过程中可以做出的选择数量有限。 目录 交互式文本模式安…

深入理解 Vue 中的指针操作(二)

文章目录 ☘️引言☘️基本用法&#x1f342;v-for指令&#x1f342;v-model指令&#x1f331;v-model适用表单控件 ☘️结论 ☘️引言 Vue.js 是一款非常流行且功能强大的前端框架&#xff0c;它以其响应式的数据绑定和组件化的开发方式赢得了众多开发者的喜爱。而在 Vue.js …

OKCC 客户中心

OKCC服务了这么多家客户中心&#xff0c;但很多小伙伴们其实并不是太了解客户中心的主要功能&#xff0c;那么我今天将从两类客户中心介绍下他们的主要功能。 一、 运营机构客户中心的功能 对于运营机构而言&#xff0c;客户中心的功能包括:能够帮助运营机构提升品牌形象&…

Chrome两个账号数据合并或者转移

文章目录 情况1情况2操作 情况1 「旧」账号&#xff1a;出于种种原因决定停用&#xff0c;但是账号里还有书签、历史记录、密码、设置等数据。 「新」账号&#xff1a;未来的主力账号。 需求是将「旧账号」的数据迁移到「新」账号之中。 情况描述&#xff1a;https://www.z…

openGauss学习笔记-134 openGauss 数据库运维-例行维护-检查操作系统参数

文章目录 openGauss学习笔记-134 openGauss 数据库运维-例行维护-检查操作系统参数134.1 检查办法134.2 异常处理 openGauss学习笔记-134 openGauss 数据库运维-例行维护-检查操作系统参数 134.1 检查办法 通过openGauss提供的gs_checkos工具可以完成操作系统状态检查。 前提…