序言
在对日期的处理过程中,如果数据库中存的日期格式和代码中需要的日期格式不同,就需要涉及到日期格式的转换,自己也经常忘记是怎么处理的。所以特此记录。
一、String与Date(java.util.Date)互转
数据库存的是Date需要转为String,或数据库存的是String,需要转为Date
1.1 String -> Date
String dateStr = "2020/08/05 10:54:23";
Date date = new Date();
//注意format的格式要与日期String的格式相匹配
DateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
try {
date = sdf.parse(dateStr);
System.out.println(date.toString());
} catch (Exception e) {
e.printStackTrace();
}
1.2 Date -> String
日期向字符串转换,可以设置任意的转换格式format
String dateStr = "";
Date date = new Date();
//format的格式可以任意
DateFormat sdf = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
DateFormat sdf2 = new SimpleDateFormat("yyyy-MM-dd HH/mm/ss");
try {
dateStr = sdf.format(date);
System.out.println(dateStr);
dateStr = sdf2.format(date);
System.out.println(dateStr);
} catch (Exception e) {
e.printStackTrace();
}
二、String与Timestamp互转
数据库存的是Timestamp需要转为String,或数据库存的是String,需要转为Timestamp
2.1 String ->Timestamp
使用Timestamp的valueOf()方法
Timestamp ts = new Timestamp(System.currentTimeMillis());
String tsStr = "2020-08-05 10:49:45";
try {
ts = Timestamp.valueOf(tsStr);
System.out.println(ts);
} catch (Exception e) {
e.printStackTrace();
}
注:String的类型必须形如: yyyy-mm-dd hh:mm:ss[.f…] 这样的格式,中括号表示可选,否则报错!!!
如果String为其他格式,可考虑重新解析下字符串,再重组~~
2.2 Timestamp -> String
使用Timestamp的toString()方法或者借用DateFormat
Timestamp ts = new Timestamp(System.currentTimeMillis());
String tsStr = "";
DateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
try {
//方法一
tsStr = sdf.format(ts);
System.out.println(tsStr);
//方法二
tsStr = ts.toString();
System.out.println(tsStr);
} catch (Exception e) {
e.printStackTrace();
}
很容易能够看出来,方法一的优势在于可以灵活的设置字符串的形式。
三、Date( java.util.Date )和Timestamp互转
声明:查API可知,Date和Timesta是父子类关系
3.1 Timestamp -> Date
Timestamp ts = new Timestamp(System.currentTimeMillis());
Date date = new Date();
try {
date = ts;
System.out.println(date);
} catch (Exception e) {
e.printStackTrace();
}
很简单,但是此刻date对象指向的实体却是一个Timestamp,即date拥有Date类的方法,但被覆盖的方法的执行实体在Timestamp中。
3.2 Date -> Timestamp
父类不能直接向子类转化,可借助中间的String~~~~
注:使用以下方式更简洁
Timestamp ts = new Timestamp(date.getTime());
注:以上日期的处理来源于博客
四、 日期的取值、转换、加减、比较
在旧版本 JDK 的时代,有不少代码中日期取值利用了 java.util.Date 类,但是由于 Date 类不便于实现国际化,其实从 JDK1.1 开始,就更推荐使用 java.util.Calendar 类进行时间和日期方面的处理。上面介绍了 Date 类的操作,下面整理了如何利用 Calendar 类取得现在的日期时间。
由于 Calendar 的构造器方法被 protected 修饰,所以我们会通过 API 中提供的 getInstance 方法来创建 Calendar 对象。
//有多个重载方法创建 Calendar 对象
Calendar now = Calendar.getInstance(); //默认
//指定时区和地区,也可以只输入其中一个参数
Calendar now = Calendar.getInstance(timeZone, locale);
//然后我们就可以通过该对象取得当前的各种时间参数了。
int year = now.get(Calendar.YEAR); //2020,当前年份
int month = now.get(Calendar.MONTH) + 1; //12,当前月,注意加 1
int day = now.get(Calendar.DATE); //05,当前日
Date date = now.getTime(); //直接取得一个 Date 类型的日期
//要取得其他类型的时间数据仅需修改 now.get() 内的参数,除了以上三种参数,其他常用参数如下:
Calendar.DAY_OF_MONTH //日期,和 Calendar.DATE 相同
Calendar.HOUR //12小时制的小时数
Calendar.HOUR_OF_DAY //24小时制的小时数
Calendar.MINUTE //分钟
Calendar.SECOND //秒
Calendar.DAY_OF_WEEK //周几
//除了取得时间数据,我们也可以通过 Calendar 对象设置各种时间参数。
//只设定某个字段的值
// public final void set(int field, int value)
now.set(Calendar.YEAR, 2020);
//设定年月日或者年月日时分或年月日时分秒
// public final void set(int year, int month, int date[, int hourOfDay, int minute, int second])
now.set(2020, 1, 1[, 11, 1, 1]);
//直接传入一个 Date 类型的日期
// public final void setTime(Date date)
now.set(date);
注意:
当设置了时间参数后,其他相关的数值都会重新计算,例如当你把日期设为 11 号后,周几就会作对应变化。
获得的月份加 1 才是实际月份。
在 Calendar 类中,周日是 1,周一是 2,以此类推。
4.1 日期转换
聊完日期取值,接下来聊聊日期转换,转换一般是 Date 型日期与 String 型字符串之间的相互转换,我主要利用 java.text.SimpleDateFormat 进行转换操作。
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
try {
//日期转字符串
Calendar calendar = Calendar.getInstance();
Date date = calendar.getTime();
String dateStringParse = sdf.format(date);
//字符串转日期
String dateString = "2020-08-05 11:11:11";
Date dateParse = sdf.parse(dateString);
} catch (ParseException e) {
e.printStackTrace();
}
注意:
创建 SimpleDateFormat 对象时必须指定转换格式。
转换格式区分大小写,yyyy 代表年份,MM 代表月份,dd 代表日期,HH 代表 24 进制的小时,hh 代表 12 进制的小时,mm 代表分钟,ss 代表秒。
4.2. 日期加减
通常来说,我们会对日期做两种加减操作:
以某个日期为基准,计算其几天前/后、几年前/后,或者其他时间单位前后的日期
//根据现在时间计算
Calendar now = Calendar.getInstance();
now.add(Calendar.YEAR, 1); //现在时间的1年后
now.add(Calendar.YEAR, -1); //现在时间的1年前
//根据某个特定的时间 date (Date 型) 计算
Calendar specialDate = Calendar.getInstance();
specialDate.setTime(date); //注意在此处将 specialDate 的值改为特定日期
specialDate.add(Calendar.YEAR, 1); //特定时间的1年后
specialDate.add(Calendar.YEAR, -1); //特定时间的1年前
注意使用了 Calendar 对象的 add 方法,可以更改 Calendar.YEAR 为任意时间单位字段,完成各种时间单位下的日期计算。
计算两个时间的间隔,例如计算 2016 年 1 月 1 日距离现在有多少天。
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String dateString = "2016-01-01 11:11:11";
Calendar calendar = Calendar.getInstance();
long nowDate = calendar.getTime().getTime(); //Date.getTime() 获得毫秒型日期
try {
long specialDate = sdf.parse(dateString).getTime();
//计算间隔多少天,则除以毫秒到天的转换公式
long betweenDate = (specialDate - nowDate) / (1000 * 60 * 60 * 24);
System.out.print(betweenDate);
} catch (ParseException e) {
e.printStackTrace();
}
4.3 日期比较
翻看自己以前的代码,发现每当进行日期比较的操作时,总会先将日期转为 “yyyyMMdd” 格式的字符串,再将字符串转为数值,然后比较数值大小。哈哈,一个简单的比较操作,却要写十几行代码,有点目不忍视。现在得说说正确地日期比较姿势是怎么样的。
日期比较一般有两种方法,对于 java.util.Date 或者 java.util.Calendar 都是通用的。一种是通过 after() 与 before() 方法进行比较,一种是通过 compareTo() 方法进行比较。
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String dateString_01 = "2020-08-05 11:11:11";
String dateString_02 = "2020-08-05 11:11:11";
try {
Date date_01 = sdf.parse(dateString_01);
Date date_02 = sdf.parse(dateString_02);
System.out.println(date_01.before(date_02)); //true,当 date_01 小于 date_02 时,为 true,否则为 false
System.out.println(date_02.after(date_01)); //true,当 date_02 大于 date_01 时,为 true,否则为 false
System.out.println(date_01.compareTo(date_02)); //-1,当 date_01 小于 date_02 时,为 -1
System.out.println(date_02.compareTo(date_01)); //1,当 date_02 大于 date_01 时,为 1
System.out.println(date_02.compareTo(date_02)); //0,当两个日期相等时,为 0
} catch (ParseException e) {
e.printStackTrace();
}
4.4. 工具库推荐
看见有朋友向我推荐了 Joda-Time 库,经过研究后发现能够简化不少 Java 中日期时间的操作,特别是在某些应用场景下还得 Date 转 Calendar 或者 String 转 Date 之类的操作。库的使用逻辑与原生十分相像,学习曲线还是比较平缓的,在此也向各位读者提供一个更加多元的选择。
Joda-Time 官方网站:http://www.joda.org/joda-time/
注:本节的内容来源于 博客原文
五、jkd8对日期的处理
package com.test.utils;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
public class DateTimeUtil {
private static final String YYYY_MM = "yyyy-MM";
private static final String YYYY_MM_DD = "yyyy-MM-dd";
private static final String YYYY_MM_DD_HH_MM_SS = "yyyy-MM-dd HH:mm:ss";
private static final String YYYYMMDDHHMM = "yyyy.MM.dd";
/**
* 获取指定日期时间的毫秒数
* @return 时间
*/
public static long getTimestampOfDateTime() {
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern(YYYY_MM_DD);
String month = LocalDateTime.now().format(dateTimeFormatter);
String time = month + " 00:00:00";
return getSeconds(time);
}
/**
* 在本月的基础上减去3月的时间,例如当前时间是2020-08-05 00:00:00,减去3个月的时间就是2020-05-05 00:00:00的毫秒数
* @return
*/
public static long getBeforeTime4(){
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd");
LocalDateTime localDateTime = LocalDateTime.now();
String time = localDateTime.minusMonths(3L).format(dateTimeFormatter) + " 00:00:00";
return getSeconds(time);
}
/**
* 获取指定日期时间的毫秒数
* @return 时间
*/
public static long getSeconds(String time){
DateTimeFormatter df = DateTimeFormatter.ofPattern(YYYY_MM_DD_HH_MM_SS);
LocalDateTime localDateTime = LocalDateTime.parse(time, df);
ZoneId zone = ZoneId.systemDefault();
Instant instant = localDateTime.atZone(zone).toInstant();
return instant.toEpochMilli()/1000;
}
/**
* 获取当天的指定时间
* @return 时间
*/
public static String getCrontabDateTime(){
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern(YYYYMMDDHHMM);
String day = LocalDateTime.now().format(dateTimeFormatter) + " 01:00:00";
return day;
}
/**
* 获取当天所在的月份
* @return 时间
*/
public static String getDateMonth(){
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern(YYYY_MM);
return LocalDateTime.now().format(dateTimeFormatter);
}
}
注:本节的内容来源于我自己写的对日期的处理
补充
另外我补充一下个人认为写的比较好的jdk8对日期的处理的博客:
1、JDK8日期处理API(转)
2、Jdk8 日期和时间处理类
3、Java 8中对日期和时间的处理(使用API处理)