带缓存的基于DateTimeFormatter的日期格式化工具类
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的日期格式化工具类的更多相关文章
- Java 日期格式化工具类
import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; impor ...
- java基础篇 -- 常用的日期加减和日期格式化工具类
平时我们遇到日期的加减,感觉是相当麻烦的,以下是常用的日志加减的方法,包括日的加减.月的加减等,也包括了一些常用的日期格式化,这样在我们以后碰到日期加减的时候会省去很多麻烦,欢迎大神指正和吐槽: pa ...
- Java ——日期时间 日期时间相关类 随机数 定义类属性时建议使用引用数据类型
本节重点思维导图 Date对象创建 两个构造函数: Date() ----使用当前日期和时间来初始化对象 Date(long millisec) -----接收一个参数,该参数是从1970年1月1日起 ...
- 基于AFN封装的带缓存的网络请求
给大家分享一个基于AFN封装的网络请求 git: https://github.com/zhouxihi/NVNetworking #带缓存机制的网络请求 各类请求有分带缓存 , 不带缓存, 可自定义 ...
- 基于Java8的日期时间工具类DateTimeFormatter
原文:https://blog.csdn.net/qq_36596145/article/details/85331002 import java.time.Instant; import java. ...
- Delphi中带缓存的数据更新技术
一. 概念 在网络环境下,数据库应用程序是c/s或者是多层结构的模式.在这种环境下,数据库应用程序的开发应当尽可能考虑减少网络数据传输量,并且尽量提高并发度.基于这个目的,带缓存的数据更新技术应运而生 ...
- JS 日期工具类-基于yDate
源码下载 前言:最近在用到JS日期操作时,发现有一些不方便,于是搜素了一些网上的资料,基于一个开源工具类-yDate 进行了个性化定制,封装成了一个日期工具类工具函数大概介绍:1.普通的日期操作2. ...
- 带缓存的输入输出-bufferedinputstream类与bufferedoutputstream类
package hengzhe.cn.o1; import java.io.*; /* * 带缓存的输入输出-bufferedinputstream类与bufferedoutputstream类 * ...
- 不带缓存的I/O和标准(带缓存的)I/O
首先,先稍微了解系统调用的概念: 系统调用,英文名system call,每个操作系统都在内核里有一些内建的函数库,这些函数可以用来完成一些系统系统调用把应用程序的请求传给内核,调用相应的 ...
随机推荐
- redis-存储命令
一.String类型: 1.赋值/取值 set key valueget key 2.设置/获取多个键值 mset key1 value1 key2 value2 … mget key1 ke ...
- 【数据库】数据库入门(三): SQL
SQL: 结构化查询语言(Structured Query Language) SQL 是由 IBM 公司首先开发产生,它是关系型数据库最早出现的商用语言之一.1974年,IBM 公司 San Jos ...
- 大数据技术之Sqoop
大数据技术之Sqoop 一.Sqoop简介 Apache Sqoop(TM)是一种旨在有效地在Apache Hadoop和诸如关系数据库等结构化数据存储之间传输大量数据的工具. Sqoop于2012 ...
- Linux命令ping
原文 ping命令用来测试主机之间网络的连通性.执行ping指令会使用ICMP传输协议,发出要求回应的信息,若远端主机的网络功能没有问题,就会回应该信息,因而得知该主机运作正常. 语法 ping(选项 ...
- 【Swagger2】解决swagger文档地址请求404的问题
一.出现的问题背景 某个项目,本地启动后,访问swagger文档地址可以访问到, http://localhost:8203/doc.html.但是部署到开发环境就访问不到,报404资源找不到的问题 ...
- pid 控制
static std::map<pid_t, TTask *> Tasks; TError TTask::Fork(bool detach) { PORTO_ASSERT(!PostFor ...
- Caused by: java.lang.IllegalStateException: duplicate key: datasource
java.lang.IllegalStateException: Failed to load property source from location 'classpath:/applicatio ...
- discuz网站前端代码优化思路
一.head标签中的局部 1.URL设计 URL尽量含有通用已成趋向的移动命名,例如“m./wap./3g./mobi./mobile./mob/wml/”,能够在子域名等方面表现 2.页面顶部的do ...
- PID动图——很形象
p是控制现在,i是纠正曾经,d是管控未来! pid的公式: 其中Kp为比例带,TI为积分时间,TD为微分时间.PID控制的基本原理就是如此. pid的原理和代码,在木南创智的博客园中有很好的教程:ht ...
- 轻松学习之三——IMP指针的作用
http://www.jianshu.com/p/425a39d43d16 可能大家一直看到有许多朋友在Runtime相关文章中介绍IMP指针的概念,那么IMP究竟有什么实际作用呢?让我们先从一个函数 ...