2024-08-20 00:42:11 +08:00

670 lines
22 KiB
Java

package xyz.zhouxy.plusone.commons.util;
import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.Month;
import java.time.YearMonth;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Calendar;
import java.util.Date;
import java.util.TimeZone;
import org.apache.commons.lang3.StringUtils;
import com.google.common.base.Preconditions;
import xyz.zhouxy.plusone.commons.collection.SafeConcurrentHashMap;
import xyz.zhouxy.plusone.commons.base.Quarter;
import xyz.zhouxy.plusone.commons.base.YearQuarter;
import xyz.zhouxy.plusone.commons.collection.MapWrapper;
/**
* 日期时间工具类
*
* @author zhouxy
*/
public class DateTimeTools {
/**
* 缓存时间格式化器
*/
private static final MapWrapper<String, DateTimeFormatter> DATE_TIME_FORMATTER_CACHE = MapWrapper
.<String, DateTimeFormatter>wrap(new SafeConcurrentHashMap<>())
.keyChecker(pattern -> Preconditions.checkArgument(StringUtils.isNotBlank(pattern), "The pattern could not be blank."))
.valueChecker(formatter -> Preconditions.checkNotNull(formatter, "The formatter could not be null."))
.build();
/**
* 获取时间格式化器
*
* @param pattern 时间格式
* @return 时间格式化器
*/
public static DateTimeFormatter getDateTimeFormatter(String pattern) {
return DATE_TIME_FORMATTER_CACHE.computeIfAbsent(pattern, DateTimeFormatter::ofPattern);
}
/**
* 将日期时间转换为指定格式的字符串
*
* @param pattern 时间格式
* @param dateTime 日期时间
* @return 格式化的字符串
*/
public static String toString(String pattern, ZonedDateTime dateTime) {
return getDateTimeFormatter(pattern).format(dateTime);
}
/**
* 将时间戳转换为指定格式的字符串,使用系统默认时区
*
* @param pattern 时间格式
* @param instant 时间戳
* @return 格式化的字符串
*/
public static String toString(String pattern, Instant instant) {
ZonedDateTime dateTime = instant.atZone(ZoneId.systemDefault());
return toString(pattern, dateTime);
}
/**
* 将时间戳转换为指定格式的字符串,使用指定时区
*
* @param pattern 时间格式
* @param instant 时间戳
* @param zone 时区
* @return 格式化的字符串
*/
public static String toString(String pattern, Instant instant, ZoneId zone) {
ZonedDateTime dateTime = instant.atZone(zone);
return toString(pattern, dateTime);
}
/**
* 指定格式,返回当前时间戳对应的字符串
*
* @param pattern 时间格式
* @return 格式化的字符串
*/
public static String nowStr(String pattern) {
return toString(pattern, ZonedDateTime.now());
}
/**
* 指定格式,返回当前时间戳对应的字符串,使用指定时区
*
* @param pattern 时间格式
* @param zone 时区
* @return 格式化的字符串
*/
public static String nowStr(String pattern, ZoneId zone) {
return toString(pattern, Instant.now().atZone(zone));
}
// toDate
/**
* 将时间戳转换为 {@link Date} 对象
*
* @param timeMillis 时间戳
* @return {@link Date} 对象
*/
public static Date toDate(long timeMillis) {
return Date.from(Instant.ofEpochMilli(timeMillis));
}
/**
* 将 {@link Calendar} 对象转换为 {@link Date} 对象
*
* @param calendar {@link Calendar} 对象
* @return {@link Date} 对象
*/
public static Date toDate(Calendar calendar) {
return calendar.getTime();
}
/**
* 将 {@link Instant} 对象转换为 {@link Date} 对象
*
* @param instant {@link Instant} 对象
* @return {@link Date} 对象
*/
public static Date toDate(Instant instant) {
return Date.from(instant);
}
/**
* 将 {@link ZonedDateTime} 对象转换为 {@link Date} 对象
*
* @param zonedDateTime {@link ZonedDateTime} 对象
* @return {@link Date} 对象
*/
public static Date toDate(ZonedDateTime zonedDateTime) {
return Date.from(zonedDateTime.toInstant());
}
/**
* 使用指定时区,将 {@link LocalDateTime} 对象转换为 {@link Date} 对象
*
* @param localDateTime {@link LocalDateTime} 对象
* @param zone 时区
* @return {@link Date} 对象
*/
public static Date toDate(LocalDateTime localDateTime, ZoneId zone) {
return Date.from(ZonedDateTime.of(localDateTime, zone).toInstant());
}
/**
* 使用指定时区,将 {@link LocalDate} 和 {@link LocalTime} 对象转换为 {@link Date} 对象
*
* @param localDate {@link LocalDate} 对象
* @param localTime {@link LocalTime} 对象
* @param zone 时区
* @return {@link Date} 对象
*/
public static Date toDate(LocalDate localDate, LocalTime localTime, ZoneId zone) {
return Date.from(ZonedDateTime.of(localDate, localTime, zone).toInstant());
}
// toInstant
/**
* 将时间戳转换为 {@link Instant} 对象
*
* @param timeMillis 时间戳
* @return {@link Instant} 对象
*/
public static Instant toInstant(long timeMillis) {
return Instant.ofEpochMilli(timeMillis);
}
/**
* 将 {@link Date} 对象转换为 {@link Instant} 对象
*
* @param date {@link Date} 对象
* @return {@link Instant} 对象
*/
public static Instant toInstant(Date date) {
return date.toInstant();
}
/**
* 将 {@link Calendar} 对象转换为 {@link Instant} 对象
*
* @param calendar {@link Calendar} 对象
* @return {@link Instant} 对象
*/
public static Instant toInstant(Calendar calendar) {
return calendar.toInstant();
}
/**
* 将 {@link ZonedDateTime} 对象转换为 {@link Instant} 对象
*
* @param zonedDateTime {@link ZonedDateTime} 对象
* @return {@link Instant} 对象
* @deprecated 请使用 {@link ZonedDateTime#toInstant()} 方法
*/
@Deprecated
public static Instant toInstant(ZonedDateTime zonedDateTime) { // NOSONAR
return zonedDateTime.toInstant();
}
/**
* 使用指定时区,将 {@link LocalDateTime} 对象转换为 {@link Instant} 对象
*
* @param LocalDateTime {@link LocalDateTime} 对象
* @param zone 时区
* @return {@link Instant} 对象
*/
public static Instant toInstant(LocalDateTime localDateTime, ZoneId zone) {
return ZonedDateTime.of(localDateTime, zone).toInstant();
}
// toZonedDateTime
/**
* 获取时间戳在指定时区的地区时间。
* <p>
* 传入不同 {@link ZoneId},获取到的 {@link ZonedDateTime} 对象实际上还是同一时间戳,
* 只是不同时区的表示。
* </p>
*
* @param timeMillis 时间戳
* @param zone 时区
* @return 带时区信息的地区时间
*/
public static ZonedDateTime toZonedDateTime(long timeMillis, ZoneId zone) {
return ZonedDateTime.ofInstant(Instant.ofEpochMilli(timeMillis), zone);
}
/**
* 获取 {@link Date} 所表示的时间戳,在指定时区的地区时间。
* <p>
* 传入不同 {@link ZoneId},获取到的 {@link ZonedDateTime} 对象实际上还是同一时间戳,
* 只是不同时区的表示。
* </p>
*
* @param dateTime {@link Date} 对象
* @param zone 时区
* @return 带时区信息的地区时间
*/
public static ZonedDateTime toZonedDateTime(Date dateTime, ZoneId zone) {
return ZonedDateTime.ofInstant(dateTime.toInstant(), zone);
}
/**
* 获取 {@link Date} 所表示的时间戳,在指定时区的地区时间。
* <p>
* 传入不同 {@link ZoneId},获取到的 {@link ZonedDateTime} 对象实际上表示的还是还是同一时间戳的时间,
* 只是不同时区的表示。
* </p>
*
* @param dateTime {@link Date} 对象
* @param timeZone 时区
* @return 带时区信息的地区时间
*/
public static ZonedDateTime toZonedDateTime(Date dateTime, TimeZone timeZone) {
return ZonedDateTime.ofInstant(dateTime.toInstant(), timeZone.toZoneId());
}
/**
* 使用 {@code calendar} 对象的时区信息,将 {@link Calendar} 对象转换为 {@link ZonedDateTime}
* 对象。
*
* @param calendar{@link Calendar} 对象
* @return {@link ZonedDateTime} 对象
*/
public static ZonedDateTime toZonedDateTime(Calendar calendar) {
return calendar.toInstant().atZone(calendar.getTimeZone().toZoneId());
}
/**
* 使用指定的时区,将 {@link Calendar} 对象转换为 {@link ZonedDateTime} 对象。
*
* @param calendar {@link Calendar} 对象
* @param zone 时区
* @return {@link ZonedDateTime} 对象
*/
public static ZonedDateTime toZonedDateTime(Calendar calendar, ZoneId zone) {
return calendar.toInstant().atZone(zone);
}
/**
* 使用指定的时区,将 {@link Calendar} 对象转换为 {@link ZonedDateTime} 对象。
*
* @param calendar {@link Calendar} 对象
* @param zone 时区
* @return {@link ZonedDateTime} 对象
*/
public static ZonedDateTime toZonedDateTime(Calendar calendar, TimeZone zone) {
return calendar.toInstant().atZone(zone.toZoneId());
}
/**
* 创建带时区的地区时间
*
* @param localDateTime 地区时间
* @param zone 时区
* @return 带时区的地区时间
*
* @deprecated 使用 {@link ZonedDateTime#of(LocalDateTime, ZoneId)}
*/
@Deprecated
public static ZonedDateTime toZonedDateTime(LocalDateTime localDateTime, ZoneId zone) { // NOSONAR
return ZonedDateTime.of(localDateTime, zone);
}
// toLocalDateTime
/**
* 获取时间戳在指定时区的地区时间。
*
* @param timeMillis 时间戳
* @param zone 时区
* @return 地区时间
*/
public static LocalDateTime toLocalDateTime(long timeMillis, ZoneId zone) {
return LocalDateTime.ofInstant(Instant.ofEpochMilli(timeMillis), zone);
}
/**
* 获取 {@link Date} 所表示的时间戳,在指定时区的地区时间。
*
* @param dateTime {@link Date} 对象
* @param zone 时区
* @return 地区时间
*/
public static LocalDateTime toLocalDateTime(Date dateTime, ZoneId zone) {
return LocalDateTime.ofInstant(dateTime.toInstant(), zone);
}
/**
* 获取 {@link Date} 所表示的时间戳,在指定时区的地区时间。
*
* @param dateTime {@link Date} 对象
* @param timeZone 时区
* @return 地区时间
*/
public static LocalDateTime toLocalDateTime(Date dateTime, TimeZone timeZone) {
return LocalDateTime.ofInstant(dateTime.toInstant(), timeZone.toZoneId());
}
/**
* 获取 {@link Calendar} 所表示的时间戳,在指定时区的地区时间。
*
* @param calendar {@link Calendar} 对象
* @param zone 时区
* @return 地区时间
*/
public static LocalDateTime toLocalDateTime(Calendar calendar, ZoneId zone) {
return LocalDateTime.ofInstant(calendar.toInstant(), zone);
}
/**
* 获取 {@link Calendar} 所表示的时间戳,在指定时区的地区时间。
*
* @param calendar {@link Calendar} 对象
* @param zone 时区
* @return 地区时间
*/
public static LocalDateTime toLocalDateTime(Calendar calendar, TimeZone zone) {
return LocalDateTime.ofInstant(calendar.toInstant(), zone.toZoneId());
}
/**
* 获取 {@link ZonedDateTime} 所表示的时间戳,在指定时区的地区时间。
*
* @param zonedDateTime {@link ZonedDateTime} 对象
* @param zone 时区
* @return 地区时间
*/
public static LocalDateTime toLocalDateTime(ZonedDateTime zonedDateTime, ZoneId zone) {
return LocalDateTime.ofInstant(zonedDateTime.toInstant(), zone);
}
// ====================
// toJodaInstant
/**
* 将 {@link java.time.Instant} 转换为 {@link org.joda.time.Instant}
*
* @param instant {@link java.time.Instant} 对象
* @return {@link org.joda.time.Instant} 对象
*/
public static org.joda.time.Instant toJodaInstant(java.time.Instant instant) {
return new org.joda.time.Instant(instant.toEpochMilli());
}
/**
* 将 {@link java.time.ZonedDateTime} 转换为 {@link org.joda.time.Instant}
*
* @param zonedDateTime {@link java.time.ZonedDateTime} 对象
* @return {@link org.joda.time.Instant} 对象
*/
public static org.joda.time.Instant toJodaInstant(java.time.ZonedDateTime zonedDateTime) {
return toJodaInstant(zonedDateTime.toInstant());
}
/**
* 计算指定时区的地区时间,对应的时间戳。结果为 {@link org.joda.time.Instant} 对象
*
* @param localDateTime {@link java.time.LocalDateTime} 对象
* @param zone 时区
* @return {@link org.joda.time.Instant} 对象
*/
public static org.joda.time.Instant toJodaInstant(java.time.LocalDateTime localDateTime, java.time.ZoneId zone) {
return toJodaInstant(java.time.ZonedDateTime.of(localDateTime, zone));
}
// toJavaInstant
/**
* 将 {@link org.joda.time.Instant} 对象转换为 {@link java.time.Instant} 对象
*
* @param instant {@link org.joda.time.Instant} 对象
* @return {@link java.time.Instant} 对象
*/
public static java.time.Instant toJavaInstant(org.joda.time.Instant instant) {
return toInstant(instant.getMillis());
}
/**
* 将 joda-time 中的 {@link org.joda.time.DateTime} 对象转换为 Java 的
* {@link java.time.Instant} 对象
*
* @param dateTime joda-time 中表示日期时间的 {@link org.joda.time.DateTime} 对象
* @return Java 表示时间戳的 {@link java.time.Instant} 对象
*/
public static java.time.Instant toJavaInstant(org.joda.time.DateTime dateTime) {
return toInstant(dateTime.getMillis());
}
/**
* 将 joda-time 中的 {@link org.joda.time.LocalDateTime} 对象和
* {@link org.joda.time.DateTimeZone} 对象
* 转换为 Java 中的 {@link java.time.Instant} 对象
*
* @param localDateTime
* @param zone
* @return
*/
public static java.time.Instant toJavaInstant(
org.joda.time.LocalDateTime localDateTime,
org.joda.time.DateTimeZone zone) {
return toJavaInstant(localDateTime.toDateTime(zone));
}
// toJodaDateTime
/**
* 将 Java 中表示日期时间的 {@link java.time.ZonedDateTime} 对象
* 转换为 joda-time 的 {@link org.joda.time.DateTime} 对象
*
* @param zonedDateTime 日期时间
* @return joda-time 中对应的 {@link org.joda.time.DateTime} 对象
*/
public static org.joda.time.DateTime toJodaDateTime(java.time.ZonedDateTime zonedDateTime) {
org.joda.time.DateTimeZone zone = org.joda.time.DateTimeZone.forID(zonedDateTime.getZone().getId());
return toJodaInstant(zonedDateTime.toInstant()).toDateTime(zone);
}
/**
* 将 java.time 中表示日期时间的 {@link java.time.LocalDateTime} 对象和表示时区的
* {@link java.time.ZoneId} 对象转换为 joda-time 中对应的 {@link org.joda.time.DateTime}
* 对象
* 转换为 joda-time 中对应的 {@link org.joda.time.DateTime} 对象
*
* @param localDateTime 日期时间
* @param zone 时区
* @return joda-time 中对应的 {@link org.joda.time.DateTime} 对象
*/
public static org.joda.time.DateTime toJodaDateTime(
java.time.LocalDateTime localDateTime,
java.time.ZoneId zone) {
org.joda.time.DateTimeZone dateTimeZone = toJodaTime(zone);
return toJodaInstant(ZonedDateTime.of(localDateTime, zone).toInstant()).toDateTime(dateTimeZone);
}
/**
* 计算时间戳在指定时区对应的时间,结果使用 {@link org.joda.time.DateTime} 表示
*
* @param instant java.time 中的时间戳
* @param zone java.time 中的时区
* @return joda-time 中带时区的日期时间
*/
public static org.joda.time.DateTime toJodaDateTime(
java.time.Instant instant,
java.time.ZoneId zone) {
org.joda.time.DateTimeZone dateTimeZone = toJodaTime(zone);
return toJodaInstant(instant).toDateTime(dateTimeZone);
}
// toZonedDateTime
/**
* 将 joda-time 中带时区的日期时间,转换为 java.time 中带时区的日期时间
*
* @param dateTime joda-time 中带时区的日期时间
* @return java.time 中带时区的日期时间
*/
public static java.time.ZonedDateTime toZonedDateTime(org.joda.time.DateTime dateTime) {
java.time.ZoneId zone = dateTime.getZone().toTimeZone().toZoneId();
return toJavaInstant(dateTime.toInstant()).atZone(zone);
}
/**
* 将 joda-time 中的 {@link org.joda.time.LocalDateTime} 和
* {@link org.joda.time.DateTimeZone}
* 转换为 java.time 中的 {@link java.time.ZonedDateTime}
*
* @param localDateTime joda-time 中的地区时间
* @param dateTimeZone joda-time 中的时区
* @return java.time 中带时区的日期时间
*/
public static java.time.ZonedDateTime toZonedDateTime(
org.joda.time.LocalDateTime localDateTime,
org.joda.time.DateTimeZone dateTimeZone) {
java.time.ZoneId zone = toJavaZone(dateTimeZone);
return toJavaInstant(localDateTime, dateTimeZone).atZone(zone);
}
/**
* 获取 joda-time 中的 {@link org.joda.time.Instant} 在指定时区的时间,用 Java 8
* {@link java.time.ZonedDateTime} 表示
*
* @param instant joda-time 中的时间戳
* @param dateTimeZone joda-time 中的时区
* @return
*/
public static java.time.ZonedDateTime toZonedDateTime(
org.joda.time.Instant instant,
org.joda.time.DateTimeZone dateTimeZone) {
java.time.ZoneId zone = toJavaZone(dateTimeZone);
return toJavaInstant(instant).atZone(zone);
}
// toJodaLocalDateTime
/**
* 将 {@link java.time.LocalDateTime} 转换为 {@link org.joda.time.LocalDateTime}
*
* @param localDateTime Java 8 LocalDateTime
* @return joda-time LocalDateTime
*/
public static org.joda.time.LocalDateTime toJodaLocalDateTime(java.time.LocalDateTime localDateTime) {
java.time.ZoneId javaZone = java.time.ZoneId.systemDefault();
org.joda.time.DateTimeZone jodaZone = toJodaTime(javaZone);
return toJodaInstant(localDateTime, javaZone).toDateTime(jodaZone).toLocalDateTime();
}
// toJavaLocalDateTime
/**
* 将 {@link org.joda.time.LocalDateTime} 转换为 {@link java.time.LocalDateTime}
*
* @param localDateTime joda-time LocalDateTime
* @return Java 8 LocalDateTime
*/
public static java.time.LocalDateTime toJavaLocalDateTime(org.joda.time.LocalDateTime localDateTime) {
org.joda.time.DateTimeZone jodaZone = org.joda.time.DateTimeZone.getDefault();
java.time.ZoneId javaZone = toJavaZone(jodaZone);
return toJavaInstant(localDateTime, jodaZone).atZone(javaZone).toLocalDateTime();
}
/**
* 转换 Java API 和 joda-time API 表示时区的对象
*
* @param jodaZone joda-time API 中表示时区的对象
* @return Java API 中表示时区的对象
*/
public static java.time.ZoneId toJavaZone(org.joda.time.DateTimeZone jodaZone) {
return jodaZone.toTimeZone().toZoneId();
}
/**
* 转换 Java API 和 joda-time API 表示时区的对象
*
* @param zone Java API 中表示时区的对象
* @return joda-time API 中表示时区的对象
*/
public static org.joda.time.DateTimeZone toJodaTime(java.time.ZoneId zone) {
return org.joda.time.DateTimeZone.forID(zone.getId());
}
// getQuarter
/**
* 获取指定日期所在季度
*
* @param date 日期
* @return 日期所在的季度
*/
public static YearQuarter getQuarter(Date date) {
return YearQuarter.of(date);
}
/**
* 获取指定日期所在季度
*
* @param date 日期
* @return 日期所在的季度
*/
public static YearQuarter getQuarter(Calendar date) {
return YearQuarter.of(date);
}
/**
* 获取指定月份所在季度
*
* @param month 月份
* @return 季度
*/
public static Quarter getQuarter(Month month) {
return Quarter.fromMonth(month);
}
/**
* 获取指定年月所在季度
*
* @param year 年
* @param month 月
* @return 季度
*/
public static YearQuarter getQuarter(int year, Month month) {
return YearQuarter.of(YearMonth.of(year, month));
}
/**
* 获取指定年月所在季度
*
* @param yearMonth 年月
* @return 季度
*/
public static YearQuarter getQuarter(YearMonth yearMonth) {
return YearQuarter.of(yearMonth);
}
/**
* 获取指定日期所在季度
*
* @param date 日期
* @return 日期所在的季度
*/
public static YearQuarter getQuarter(LocalDate date) {
return YearQuarter.of(date);
}
/**
* 私有构造方法,明确标识该常量类的作用。
*/
private DateTimeTools() {
throw new IllegalStateException("Utility class");
}
}