Java8中为月日新增了类MonthDay,可以用来处理生日,节日、纪念日和星座等周期性问题。
1.MonthDay
特别需要注意的:它的默认打印格式会带前缀"--" ,比如--12-03,同样的默认解析格式也需要加前缀。
1.1 部分源码
* @implSpec
* This class is immutable and thread-safe.
*
* @since 1.8
*/
public final class MonthDay
implements TemporalAccessor, TemporalAdjuster, Comparable<MonthDay>, Serializable {
/**
* Serialization version.
*/
private static final long serialVersionUID = -939150713474957432L;
/**
* Parser.
*/
private static final DateTimeFormatter PARSER = new DateTimeFormatterBuilder()
.appendLiteral("--")
.appendValue(MONTH_OF_YEAR, 2)
.appendLiteral('-')
.appendValue(DAY_OF_MONTH, 2)
.toFormatter();
/**
* The month-of-year, not null.
*/
private final int month;
/**
* The day-of-month.
*/
private final int day;
通过源码可以看出使用final修饰MonthDay,MonthDay是线程安全类,同时实现了TemporalAccessor, TemporalAdjuster, Comparable<MonthDay>, Serializable接口,有属性读取和设置等功能,但由于没有年部分,闰年2月29日的原因,没有加减功能。
1.2 创建方式
MonthDay monthDay1 = MonthDay.now();
System.out.println(monthDay1);
MonthDay monthDay2 = MonthDay.of(12, 3);
System.out.println(monthDay2);
输出:
--02-29
--12-03
1.3 解析
System.out.println(MonthDay.parse("--12-03"));
2. 应用
对比相同月日和推算等
2.1 应用代码
/**
* 相同月日比较判断,用于生日,节日等周期性的日期比较判断。
* @param localDate1
* @param monthDay
* @return
*/
public static boolean isSameMonthDay(LocalDate localDate1, MonthDay monthDay){
Objects.requireNonNull(localDate1, "localDate1");
Objects.requireNonNull(monthDay, "monthDay");
return MonthDay.of(localDate1.getMonthValue(), localDate1.getDayOfMonth()).equals(monthDay);
}
/**
* 相同月日比较判断,用于生日,节日等周期性的日期比较判断。
* @param localDate1
* @param monthDayStr MM-dd格式
* @return
*/
public static boolean isSameMonthDay(LocalDate localDate1, String monthDayStr){
Objects.requireNonNull(monthDayStr, "monthDayStr");
return isSameMonthDay(localDate1, MonthDay.parse(MONTHDAY_FORMAT_PRE + monthDayStr));
}
/**
* 相同月日比较判断,用于生日,节日等周期性的日期比较判断。
* @param localDate1
* @param localDate2
* @return
*/
public static boolean isSameMonthDay(LocalDate localDate1, LocalDate localDate2){
Objects.requireNonNull(localDate2, "localDate2");
return isSameMonthDay(localDate1, MonthDay.of(localDate2.getMonthValue(), localDate2.getDayOfMonth()));
}
/**
* 相同月日比较判断,用于生日,节日等周期性的日期比较判断。
* @param date
* @param monthDayStr MM-dd格式
* @return
*/
public static boolean isSameMonthDay(Date date, String monthDayStr){
return isSameMonthDay(DateTimeConverterUtil.toLocalDate(date), monthDayStr);
}
/**
* 相同月日比较判断,用于生日,节日等周期性的日期比较判断。
* @param date1
* @param date2
* @return
*/
public static boolean isSameMonthDay(Date date1, Date date2){
Objects.requireNonNull(date1, "date1");
Objects.requireNonNull(date2, "date2");
return isSameMonthDay(DateTimeConverterUtil.toLocalDate(date1), DateTimeConverterUtil.toLocalDate(date2));
}
/**
* 相同月日比较判断,与当前日期对比,用于生日,节日等周期性的日期比较判断
* @param monthDayStr MM-dd格式
* @return
*/
public static boolean isSameMonthDayOfNow(String monthDayStr){
return isSameMonthDay(LocalDate.now(), monthDayStr);
}
/**
* 下个固定月日相差天数,用于生日,节日等周期性的日期推算
* @param localDate1
* @param month
* @param dayOfMonth
* @return
*/
public static long betweenNextSameMonthDay(LocalDate localDate1, int month, int dayOfMonth) {
Objects.requireNonNull(localDate1, "localDate1");
MonthDay monthDay1 = MonthDay.of(localDate1.getMonthValue(), localDate1.getDayOfMonth());
MonthDay monthDay2 = MonthDay.of(month, dayOfMonth);
// localDate1 月日 小于 month dayOfMonth
if (monthDay1.compareTo(monthDay2) == -1) {
return betweenTotalDays(localDate1.atStartOfDay(),
localDate1.withMonth(month).withDayOfMonth(dayOfMonth).atStartOfDay());
} else {
// 闰年2月29
MonthDay leapMonthDay = MonthDay.of(2, 29);
if (leapMonthDay.equals(monthDay2)) {
LocalDate nextLeapYear = nextLeapYear(localDate1);
return betweenTotalDays(localDate1.atStartOfDay(),
nextLeapYear.withMonth(month).withDayOfMonth(dayOfMonth).atStartOfDay());
} else {
LocalDate next = localDate1.plusYears(1);
return betweenTotalDays(localDate1.atStartOfDay(),
next.withMonth(month).withDayOfMonth(dayOfMonth).atStartOfDay());
}
}
}
/**
* 下个固定月日相差天数,用于生日,节日等周期性的日期推算
* @param localDate
* @param monthDayStr MM-dd格式
* @return
*/
public static long betweenNextSameMonthDay(LocalDate localDate, String monthDayStr) {
Objects.requireNonNull(monthDayStr, "monthDayStr");
MonthDay monthDay2 = MonthDay.parse(MONTHDAY_FORMAT_PRE + monthDayStr);
return betweenNextSameMonthDay(localDate, monthDay2.getMonthValue(), monthDay2.getDayOfMonth());
}
/**
* 下个固定月日相差天数,用于生日,节日等周期性的日期推算
* @param date
* @param monthDayStr MM-dd格式
* @return
*/
public static long betweenNextSameMonthDay(Date date, String monthDayStr) {
Objects.requireNonNull(monthDayStr, "monthDayStr");
MonthDay monthDay2 = MonthDay.parse(MONTHDAY_FORMAT_PRE + monthDayStr);
return betweenNextSameMonthDay(DateTimeConverterUtil.toLocalDate(date), monthDay2.getMonthValue(),
monthDay2.getDayOfMonth());
}
/**
* 下个固定月日相差天数,与当前日期对比,用于生日,节日等周期性的日期推算
* @param monthDayStr MM-dd格式
* @return
*/
public static long betweenNextSameMonthDayOfNow(String monthDayStr) {
Objects.requireNonNull(monthDayStr, "monthDayStr");
MonthDay monthDay2 = MonthDay.parse(MONTHDAY_FORMAT_PRE + monthDayStr);
return betweenNextSameMonthDay(LocalDate.now(), monthDay2.getMonthValue(),
monthDay2.getDayOfMonth());
}
/**
* 下个固定月日相差日期,用于生日,节日等周期性的日期推算
* @param localDate
* @param monthDayStr MM-dd格式
* @return
*/
public static LocalDate nextSameMonthDay(LocalDate localDate, String monthDayStr){
return localDate.plusDays(betweenNextSameMonthDay(localDate, monthDayStr));
}
/**
* 下个固定月日相差日期,用于生日,节日等周期性的日期推算
* @param date
* @param monthDayStr MM-dd格式
* @return
*/
public static Date nextSameMonthDay(Date date, String monthDayStr){
return DateTimeConverterUtil.toDate(nextSameMonthDay(DateTimeConverterUtil.toLocalDate(date), monthDayStr));
}
/**
* 下个固定月日相差日期,与当前日期对比,用于生日,节日等周期性的日期推算
* @param monthDayStr MM-dd格式
* @return
*/
public static Date nextSameMonthDayOfNow(String monthDayStr){
return nextSameMonthDay(new Date(), monthDayStr);
}
2.2 测试代码
/**
* 相同月日对比
*/
@Test
public void sameMonthDayTest(){
Date date = DateTimeCalculatorUtil.getDate(2020, 2, 29);
System.out.println(date);
//date的月日部分是否和02-29相等
System.out.println(DateTimeCalculatorUtil.isSameMonthDay(date, "02-29"));
//date的月日部分是否和new Date()的月日部分相等
System.out.println(DateTimeCalculatorUtil.isSameMonthDay(date, new Date()));
//当前时间月日部分是否和02-29相等
System.out.println(DateTimeCalculatorUtil.isSameMonthDayOfNow("02-29"));
//date的月日部分和下一个03-05相差天数
System.out.println(DateTimeCalculatorUtil.betweenNextSameMonthDay(date, "03-05"));
//当前时间月日部分和下一个03-05相差天数
System.out.println(DateTimeCalculatorUtil.betweenNextSameMonthDayOfNow("03-05"));
//date为准,下一个02-14的日期
System.out.println(DateTimeCalculatorUtil.nextSameMonthDay(date, "02-14"));
//date为准,下一个03-05的日期
System.out.println(DateTimeCalculatorUtil.nextSameMonthDay(date, "03-05"));
//date为准,下一个02-29的日期 ,02-29 只有闰年有。
System.out.println(DateTimeCalculatorUtil.nextSameMonthDay(date, "02-29"));
//当前时间为准,下一个02-29的日期 ,02-29 只有闰年有。
System.out.println(DateTimeCalculatorUtil.nextSameMonthDayOfNow("02-29"));
}
2.3 输出
Sat Feb 29 00:00:00 CST 2020
true
true
true
5
5
Sun Feb 14 00:00:00 CST 2021
Thu Mar 05 00:00:00 CST 2020
Thu Feb 29 00:00:00 CST 2024
Thu Feb 29 00:00:00 CST 2024
寻找撬动地球的支点(解决问题的方案),杠杆(Java等编程语言)已经有了。