前言
日期使用问题主要是格式转换的问题
场景:通过excel导入数据,其中一个字段为出生日期,需要对字段值进行合法性校验
博客地址:
芒果橙的个人博客
【http://mangocheng.com】
文章目录
一、个人浅谈日期
时间日期作为一个基础的标识和维度,基本上所有的业务都会涉及到,因此凡是涉及到业务的数据表,都可能会加上类似创建时间、操作时间的字段。在程序开发中,前端、后端、数据库在进行数据的传递过程中,对于时间的格式就存在要求,这部分也正是问题的触发点,即格式的转换。
二、日期格式转换遇到的问题
1. 场景:通过excel导入数据,其中一个字段为出生日期,需要对字段值进行合法性校验
2. 使用过程
1)使用SimpleDateFormat进行日期转化,如果抛出异常则说明填入的出生日期有误
1. 日期的分隔符不同
try {
String birthDate = "2000-12-31";
SimpleDateFormat format = new SimpleDateFormat("yyyy/MM/dd");
format.parse(birthDate);
System.out.println("日期正常");
} catch (ParseException e) {
System.out.println("日期格式有误");
}
-
输出:日期格式有误
-
原因:格式不对,输入的分割符为
–
,转化的格式为
/
2. 非法日期
// 出生日期为12月32日
try {
String birthDate = "2000-12-32".replaceAll("-","/");
SimpleDateFormat format = new SimpleDateFormat("yyyy/MM/dd");
format.parse(birthDate);
System.out.println("日期正常");
} catch (ParseException e) {
System.out.println("日期格式有误");
e.printStackTrace();
}
- 输出:日期正常
- 原因:后面说明
-
重点:虽然是12月32日,但并不会抛出异常,当时是很奇怪的,怎么会是正常的,但我没有多想,觉得是不是SimpleDateFormat本身的转化存在缺陷,于是我想着换另一个工具类
2)使用apache的org.apache.commons.lang3.time.DateUtils
- 非法日期
try {
String birthDate = "2000-12-32".replaceAll("-", "");
SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd");
org.apache.commons.lang3.time.DateUtils.parseDate(birthDate,"yyyyMMdd");
System.out.println("日期正常");
} catch (ParseException e) {
System.out.println("日期格式有误");
}
- 输出:日期正常
- 原因:与1相同
-
我再次认为是本身工具类的问题,于是我选择了用公司封装的工具类
3)使用公司的日期工具类
- 非法日期
try {
String birthDate = "2000-12-32";
birthDate = DateUtils.formatDate(format.parse(birthDate));
System.out.println("日期正常");
} catch (ParseException e) {
System.out.println("日期格式有误");
}
- 输出:日期格式有误
- 原因:非法日期,没有12月32日
-
用公司的工具类就正常,其他第三方的工具类反而有错,于是我搜索了下,终于明白了,原来日期转换有一个属性是lenient
3. 结论:lenient属性是关键
-
lenient属性(默认为true):
日期转换有一个属性是lenient——是否宽松转换(不严格解析),即如果输入的日期不合法,但能转换为日期,在不严格解析(lenient=true)时,不会抛出异常,而是会自动计算出一个新的日期(2000-12-32转换为2001-1-1)
-
DateFormat源码:DateFormat.parse()
- 测试
-
设置严格解析:lenient=false
try { String birthDate = "2000-12-32".replaceAll("-","/"); SimpleDateFormat format = new SimpleDateFormat("yyyy/MM/dd"); // 严格解析 format.setLenient(false); Date newDate = format.parse(birthDate); System.out.println("日期正常"); System.out.println("转化后的日期:" + newDate); } catch (ParseException e) { System.out.println("日期格式有误"); }
- 输出:日期格式有误
-
SimpleDateFormat和org.apache.commons.lang3.time.DateUtils默认设置为不严格解析,而刚好公司的工具类使用的日期格式器是spring框架的DateFormatter,为严格解析
-
org.apache.commons.lang3.time.DateUtils
-
spring框架的日期格式器类
-