一、背景
- 项目中经常会涉及时间的处理及持久化,而时间是有时区和夏令时的;
- 而Web项目中,客户端(浏览器操作系统)有1套时区和夏令时,服务端(操作系统)也有,数据库还有,java也有。他们记录的时间显示出来,与用户想看到的不一定匹配;
- 基于上述2点考虑,时间必须统一到一个地方处理:就在Java后端处理,其它地方全部使用UTC跟后台交互;展示时,再转成对应时区的时间处理。
二、问题
- 解决上述问题的过程中,有这样一个业务场景:需要查询当天0时0分0秒-当天23时59分59秒的数据;
注意:
- 需要获取用户当前时区下的0时0分0秒-当天23时59分59秒,而不是UTC时间下的0时0分0秒;
三、解决方案
方案一
- 可以先获取1个今天的时间,比如System.currentTimeMillis()或者new Date(),二者等价;
- 通过时间格式化方式抹掉时分秒,不就变成了当天的0时0分0秒了吗?
Date date = new Date();
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
sdf.setTimeZone(TimeZone.getDefault());
String now = sdf.format(date);
try
{
Date newDate = sdf.parse(now);
long start1 = newDate.getTime();
System.out.println("current utc mills1:" + start1);
return start1;
}
catch (ParseException e)
{
System.out.println("error:" + e);
}
注意:
- Date对象然后按照本地时间格式化后,显示的当然是当前天的本地起始时间。比如:2019-09-30;
- 再把2019-09-30转成Date对象即可;
运行结果如下:
current utc mills1:1569772800000
方案二
使用新的java 8 sdk api(来自网络,作者不详):
//当天零点
LocalDateTime.of(LocalDate.now(),LocalTime.MIN).toEpochSecond(OffsetDateTime.now().getOffset());
四、总结
- 时间处理其实挺复杂的,尽量屏蔽掉多个处理的入口,使用统一的方式来做,比如时间处理全部放在Java后台处理,数据库使用UTC时间戳,这样就屏蔽了数据库时区的影响;
- 条条大道通罗马,尽量使用简洁的方式来做。比如项目不支持JDK8,则可以使用第1种;支持的话使用第2种会更简单。第2种的使用方式不太好理解。