JAVA中的SimpleDateFormat是非线程安全的,所有在1.8的JDK版本里提供了线程安全的DateTimeFormatter类,由于是线程安全的,故我们可以将此类缓存起来多次利用提高效率。

同时在JDK8中提供了LocalDate、LocalTime、LocalDateTime,下面的工具类也对这3个新类提供了格式化和反格式化的支持。

package com.longge.util;

import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.Period;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
import java.util.Calendar;
import java.util.Date;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit; import org.apache.commons.lang3.StringUtils; import com.longge.exception.TimeUnitNotSupportException; import lombok.NonNull; /**
* 带缓存的基于DateTimeFormatter的日期格式化工具类
* @author yangzhilong
* @date 6/21/2019
*/
public class DateUtils {
private static final Map<String, DateTimeFormatter> FORMATTER_CACHE = new ConcurrentHashMap<>(); private DateUtils() {} /**
* 日期的格式化定义
* @author yangzhilong
* @date 6/21/2019
*/
public static class Pattern {
private Pattern() {} public static class Date {
private Date() {} /**
* yyyy-MM-dd
*/
public static final String YYYY_MM_DD = "yyyy-MM-dd";
/**
* yyyy-M-dd
*/
public static final String YYYY_M_DD = "yyyy-M-dd";
/**
* yyyy-M-d
*/
public static final String YYYY_M_D = "yyyy-M-d";
/**
* yyyy/MM/dd
*/
public static final String YYYY_MM_DD_2 = "yyyy/MM/dd";
/**
* yyyy/M/dd
*/
public static final String YYYY_M_DD_2 = "yyyy/M/dd";
/**
* yyyy/M/d
*/
public static final String YYYY_M_D_2 = "yyyy/M/d";
/**
* yyyyMMdd
*/
public static final String YYYYMMDD = "yyyyMMdd";
/**
* yyyyMdd
*/
public static final String YYYYMDD = "yyyyMdd";
/**
* yyyyMd
*/
public static final String YYYYMD = "yyyyMd"; /**
* MM-dd-yyyy
*/
public static final String MM_DD_YYYY = "MM-dd-yyyy";
/**
* M-dd-yyyy
*/
public static final String M_DD_YYYY = "M-dd-yyyy";
/**
* M-d-yyyy
*/
public static final String M_D_YYYY = "M-d-yyyy";
/**
* MM/dd/yyyy
*/
public static final String MM_DD_YYYY_2 = "MM/dd/yyyy";
/**
* M/dd/yyyy
*/
public static final String M_DD_YYYY_2 = "M/dd/yyyy";
/**
* M/d/yyyy
*/
public static final String M_D_YYYY_2 = "M/d/yyyy";
/**
* MMddyyyy
*/
public static final String MMDDYYYY = "MMddyyyy";
/**
* Mddyyyy
*/
public static final String MDDYYYY = "Mddyyyy";
/**
* Mdyyyy
*/
public static final String MDYYYY = "Mdyyyy";
} public static class DateTime {
private DateTime() {} /**
* yyyy-MM-dd HH:mm:ss
*/
public static final String YYYY_MM_DD_SPACE_HH_MM_SS = "yyyy-MM-dd HH:mm:ss";
/**
* yyyy-M-d H:m:s
*/
public static final String YYYY_M_D_SPACE_H_M_S = "yyyy-M-d H:m:s";
/**
* yyyy/MM/dd HH:mm:ss
*/
public static final String YYYY_MM_DD_SPACE_HH_MM_SS2 = "yyyy/MM/dd HH:mm:ss";
/**
* HH:mm yyyy-MM-dd
*/
public static final String HH_MM_SPACE_YYYY_MM_DD = "HH:mm yyyy-MM-dd";
/**
* HH:mm yyyy/MM/dd
*/
public static final String HH_MM_SPACE_YYYY_MM_DD2 = "HH:mm yyyy/MM/dd";
/**
* H:mm yyyy-MM-dd
*/
public static final String H_MM_SPACE_YYYY_MM_DD = "H:mm yyyy-MM-dd";
/**
/**
* H:mm yyyy/MM/dd
*/
public static final String H_MM_SPACE_YYYY_MM_DD2 = "H:mm yyyy/MM/dd";
/**
* H:mm,yyyy-MM-dd
*/
public static final String H_MM_COMMA_YYYY_MM_DD = "H:mm,yyyy-MM-dd";
/**
* H:mm,yyyy-MM-dd
*/
public static final String H_MM_COMMA_YYYY_MM_DD2 = "H:mm,yyyy/MM/dd";
/**
* H:mm,yyyy-M-d
*/
public static final String H_MM_COMMA_YYYY_M_D = "H:mm,yyyy-M-d";
/**
* H:m,yyyy-M-d
*/
public static final String H_M_COMMA_YYYY_M_D = "H:m,yyyy-M-d";
/**
* H:mm,yyyy-M-d
*/
public static final String H_MM_COMMA_YYYY_M_D2 = "H:mm,yyyy/M/d";
/**
* H:m,yyyy-M-d
*/
public static final String H_M_COMMA_YYYY_M_D2 = "H:m,yyyy/M/d";
}
} /**
* 格式化不带时分秒的日期
* @param date
* @param pattern {@link DateUtils.Pattern.Date}
* @return
*/
public static String formatDate(@NonNull Date date, @NonNull String pattern) {
Instant instant = date.toInstant();
ZoneId zone = ZoneId.systemDefault();
LocalDate localDate = LocalDateTime.ofInstant(instant, zone).toLocalDate();
return formatDate(localDate, pattern);
} /**
* 格式化不带时分秒的LocalDate
* @param localDate
* @param pattern {@link DateUtils.Pattern.Date}
* @return
*/
public static String formatDate(@NonNull LocalDate localDate, @NonNull String pattern) {
if(StringUtils.isBlank(pattern)) {
throw new NullPointerException("pattern is null");
} DateTimeFormatter formatter = createCacheFormatter(pattern);
return localDate.format(formatter);
} /**
* 格式化带时分秒的日期
* @param date
* @param pattern {@link DateUtils.Pattern.DateTime}
* @return
*/
public static String formatDateTime(@NonNull Date date, @NonNull String pattern) {
LocalDateTime formatDateTime = LocalDateTime.ofInstant(date.toInstant(), ZoneId.systemDefault());
return formatDateTime(formatDateTime, pattern);
} /**
* 格式化带时分秒的LocalDateTime
* @param localDateTime
* @param pattern {@link DateUtils.Pattern.DateTime}
* @return
*/
public static String formatDateTime(@NonNull LocalDateTime localDateTime, @NonNull String pattern) {
if(StringUtils.isBlank(pattern)) {
throw new NullPointerException("pattern is null");
}
DateTimeFormatter formatter = createCacheFormatter(pattern);
return localDateTime.format(formatter);
} /**
* 转换不带时分秒的字符串到Date
* @param date
* @param pattern {@link DateUtils.Pattern.Date}
* @return
*/
public static Date parseDate(@NonNull String date, @NonNull String pattern) {
LocalDate localDate = parseLocalDate(date, pattern);
Instant instant = localDate.atStartOfDay().atZone(ZoneId.systemDefault()).toInstant();
return Date.from(instant);
} /**
* 转换不带时分秒的字符串到LocalDate
* @param date
* @param pattern {@link DateUtils.Pattern.Date}
* @return
*/
public static LocalDate parseLocalDate(@NonNull String date, @NonNull String pattern) {
if(StringUtils.isBlank(date)) {
throw new NullPointerException("date is null");
}
if(StringUtils.isBlank(pattern)) {
throw new NullPointerException("pattern is null");
}
DateTimeFormatter formatter = createCacheFormatter(pattern);
return LocalDate.parse(date, formatter);
} /**
* 转换带时分秒的字符串到Date
* @param dateTime
* @param pattern {@link DateUtils.Pattern.Date}
* @return
*/
public static Date parseDateTime(@NonNull String dateTime, @NonNull String pattern) {
if(StringUtils.isBlank(dateTime)) {
throw new NullPointerException("date is null");
}
if(StringUtils.isBlank(pattern)) {
throw new NullPointerException("pattern is null");
}
LocalDateTime localDateTime = parseLocalDateTime(dateTime, pattern);
Instant instant = localDateTime.atZone(ZoneId.systemDefault()).toInstant();
return Date.from(instant);
} /**
* 转换带时分秒的字符串到LocalDateTime
* @param dateTime
* @param pattern {@link DateUtils.Pattern.DateTime}
* @return
*/
public static LocalDateTime parseLocalDateTime(@NonNull String dateTime, @NonNull String pattern) {
DateTimeFormatter formatter = createCacheFormatter(pattern);
return LocalDateTime.parse(dateTime, formatter);
} /**
* 日期的加减(仅支持天/小时/分/秒)
* @param date
* @param addValue
* @param unit
* @return
*/
public static Date dateAdd(@NonNull Date date, int addValue, TimeUnit unit) {
Calendar cal = Calendar.getInstance();
cal.setTime(date); if (TimeUnit.DAYS.equals(unit)) {
cal.add(Calendar.DAY_OF_YEAR, addValue);
} else if (TimeUnit.HOURS.equals(unit)) {
cal.add(Calendar.HOUR_OF_DAY, addValue);
} else if (TimeUnit.MINUTES.equals(unit)) {
cal.add(Calendar.MINUTE, addValue);
} else if (TimeUnit.SECONDS.equals(unit)) {
cal.add(Calendar.SECOND, addValue);
} else {
throw new TimeUnitNotSupportException();
}
return cal.getTime();
} /**
* 计算2个日期的差
* @param begin
* @param end
* @return
*/
public static Period calculationPeriod(@NonNull Date begin, Date end) {
Calendar beginCal = Calendar.getInstance();
beginCal.setTime(begin); Calendar endCal = Calendar.getInstance();
endCal.setTime(end); LocalDate beginLocal =
LocalDate.of(beginCal.get(Calendar.YEAR), beginCal.get(Calendar.MONTH), beginCal.get(Calendar.DAY_OF_YEAR));
LocalDate endLocal =
LocalDate.of(beginCal.get(Calendar.YEAR), beginCal.get(Calendar.MONTH), beginCal.get(Calendar.DAY_OF_YEAR)); return Period.between(beginLocal, endLocal);
} /**
* DateTimeFormatter缓存处理
* @param begin
* @param end
* @return
*/
private static DateTimeFormatter createCacheFormatter(String pattern) {
if (pattern == null || pattern.length() == 0) {
throw new IllegalArgumentException("Invalid pattern specification");
} DateTimeFormatter formatter = FORMATTER_CACHE.get(pattern);
if(null == formatter) {
formatter = DateTimeFormatter.ofPattern(pattern);
FORMATTER_CACHE.put(pattern, formatter);
} return formatter;
}
}

带缓存的基于DateTimeFormatter的日期格式化工具类的更多相关文章

  1. Java 日期格式化工具类

    import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; impor ...

  2. java基础篇 -- 常用的日期加减和日期格式化工具类

    平时我们遇到日期的加减,感觉是相当麻烦的,以下是常用的日志加减的方法,包括日的加减.月的加减等,也包括了一些常用的日期格式化,这样在我们以后碰到日期加减的时候会省去很多麻烦,欢迎大神指正和吐槽: pa ...

  3. Java ——日期时间 日期时间相关类 随机数 定义类属性时建议使用引用数据类型

    本节重点思维导图 Date对象创建 两个构造函数: Date() ----使用当前日期和时间来初始化对象 Date(long millisec) -----接收一个参数,该参数是从1970年1月1日起 ...

  4. 基于AFN封装的带缓存的网络请求

    给大家分享一个基于AFN封装的网络请求 git: https://github.com/zhouxihi/NVNetworking #带缓存机制的网络请求 各类请求有分带缓存 , 不带缓存, 可自定义 ...

  5. 基于Java8的日期时间工具类DateTimeFormatter

    原文:https://blog.csdn.net/qq_36596145/article/details/85331002 import java.time.Instant; import java. ...

  6. Delphi中带缓存的数据更新技术

    一. 概念 在网络环境下,数据库应用程序是c/s或者是多层结构的模式.在这种环境下,数据库应用程序的开发应当尽可能考虑减少网络数据传输量,并且尽量提高并发度.基于这个目的,带缓存的数据更新技术应运而生 ...

  7. JS 日期工具类-基于yDate

    源码下载 前言:最近在用到JS日期操作时,发现有一些不方便,于是搜素了一些网上的资料,基于一个开源工具类-yDate 进行了个性化定制,封装成了一个日期工具类工具函数大概介绍:1.普通的日期操作2. ...

  8. 带缓存的输入输出-bufferedinputstream类与bufferedoutputstream类

    package hengzhe.cn.o1; import java.io.*; /* * 带缓存的输入输出-bufferedinputstream类与bufferedoutputstream类 * ...

  9. 不带缓存的I/O和标准(带缓存的)I/O

    首先,先稍微了解系统调用的概念:       系统调用,英文名system call,每个操作系统都在内核里有一些内建的函数库,这些函数可以用来完成一些系统系统调用把应用程序的请求传给内核,调用相应的 ...

随机推荐

  1. java web的转发与重定向

    java web的转发与重定向 原创 2012年12月15日 11:51:39 19312 转发和重定向的区别 一.使用方法 转发:request.getRequestDispatcher(" ...

  2. RxJS——订阅(Subscription)

    订阅(Subscription) 什么是订阅?订阅是一个对象,它表示一个处理完就释放(disposable)的资源,是 Observable 的一个执行程序.订阅有一个很重要的方法,unsubscri ...

  3. Centos6.5基于GPT格式磁盘分区

    1.查看分区 fdisk -l 2.设置分区类型未gpt格式. parted -s /dev/sdb mklabel gpt 3.基于ext3文件系统类型格式化. mkfs.ext3 /dev/sdb ...

  4. CentOS操作系统内核升级

    一.升级内核(带aufs模块,记住一定要升级,要不然会出现很多莫名奇怪的问题,建议用yum安装) 1.yum安装带aufs模块的3.10内核(或到这里下载kernel手动安装:http://down. ...

  5. SRVCC B1,B2事件总结

    何为SRVCC? SRVCC(Single Radio Voice Call Continuity)是3GPP提出的一种VoLTE语音业务连续性方案,主要是为了解决当单射频UE 在LTE网络和2G/3 ...

  6. Broadcast,Scatter,Gather,Reduce,All-reduce分别是什么?

    Broadcast 看名字就很好理解了,其实就是把同一份数据分发广播给所有人,示意图如下: Scatter 不同于Broadcast, scatter可以将不同数据分发给不同的进程. Gather 这 ...

  7. 软件工程1916|W(福州大学)_助教博客】助教总结

    2018年上个学期,我担任计算机专业软件工程实践课程的助教,因为有汪老师和她的两位优秀研究生担任助教,基本上没有让我做什么事,除了参加了几次课程答辩.我对于助教需要做的事情其实还是没什么概念的.这个学 ...

  8. Spring Boot 之:Actuator 监控

    在 Spring Boot 2.x 中为了安全,Actuator 只开放了两个端点 /actuator/health 和 /actuator/info.可以在配置文件中设置打开. Actuator 默 ...

  9. MapReduce的初识

    MapReduce是什么 HDFS:分布式存储系统 MapReduce:分布式计算系统 YARN:hadoop 的资源调度系统 Common:以上三大组件的底层支撑组件,主要提供基础工具包和 RPC ...

  10. python 查询文件存放地址

    import os, time import sys import re def search(path,name): for root, dirs, files in os.walk(path): ...