本文讲解如何在tomcat启动时设置JVM默认时区。
环境:JDK1.8.114
web容器:Tomcat 9
tomcat启动脚本 /etc/init.d/tomcat
操作系统ubuntu 16 amd64
作为java开发人员,以下代码应该不会觉有有什么意外,运行正常。
是的,他的确能正常运行。
String sql = "insert into `order`(clientid,licensestr,type,amount,`status`,months,info,createtime) values(?,?,?,?,?,?,?,?)"; try { PreparedStatement pst = conn.prepareStatement(sql,Statement.RETURN_GENERATED_KEYS); pst.setLong(1, clientid); pst.setString(2, String.format("SN-%s",generateNextSequence())); pst.setInt(3, 1); pst.setInt(4, 1); pst.setString(5, "status/waitforlicense"); pst.setInt(6, -1); pst.setString(7, ""); SimpleDateFormat format=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); pst.setString(8, format.format(new Date())); int effect = pst.executeUpdate(); if(effect>0){ ResultSet rs = pst.getGeneratedKeys(); if(rs.next()) { long last_inserted_id = rs.getLong(1); return last_inserted_id; } } } catch (SQLException e) { e.printStackTrace(); }
他会在数据库中插入的createtime是当前JVM的默认本地(Locale)时间。因为特殊需要,笔者曾经在非东八区时间(GMT+8) --北京时间 部署项目。
录入数据库的时间不是北京时间,当业务数据捞(load)上来的时候一头雾水,也与业务数据时间不一致,产生了很多烦恼。
有的小伙伴会想到以下命令(不同版本的Ubuntu命令不一样)
cp /usr/share/zoneinfo/Asia/ShangHai /etc/localtime 或 cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
笔者在本地虚拟机执行以上命令就能得到正确的时间,但是笔者购买的虚拟机环境已经设置了正确的时间,
Tomcat获取的时间仍旧不对,后来笔者找到了类似(jdk1.8)代码,替换以上的createtime字段,插入的数据也是对的。
ZoneId zoneId = ZoneId.of("GMT+8"); ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(Instant.now(), zoneId); System.out.println(zonedDateTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
以上这段代码来自Stackoverflow:How to get local time of different time zones?
根据ZoneId.of的api说明,你还可以选择其他更多格式,例如 America/New_York,Z,UTC+01:00,GMT,GMT-2,-4等,具体可以参考ZoneId API。
当前JVM更多时区获取方法可以参考以下代码:
String[] ids = TimeZone.getAvailableIDs(); for (String id : ids) { System.out.println(displayTimeZone(TimeZone.getTimeZone(id))); } private static String displayTimeZone(TimeZone tz) { long hours = TimeUnit.MILLISECONDS.toHours(tz.getRawOffset()); long minutes = TimeUnit.MILLISECONDS.toMinutes(tz.getRawOffset()) - TimeUnit.HOURS.toMinutes(hours); // avoid -4:-30 issue minutes = Math.abs(minutes); String result = ""; if (hours > 0) { result = String.format("(GMT+%d:%02d) %s", hours, minutes, tz.getID()); } else { result = String.format("(GMT%d:%02d) %s", hours, minutes, tz.getID()); } return result; }
有些特殊情况下是不允许修改代码的,如没有编译环境,代码闭源等。
这个时候我们就要从JVM入手了,一个tomcat进程对应一个JVM,我们可以对这个JVM设置一些启动参数,带着这个疑问我们找到了解决方法
Change time zone for tomcat
windows下编辑tomcat/bin/setclasspath.bat 在所有代码之前添加 set JAVA_OPTS="-Duser.timezone=GMT" linux编辑tomcat/bin/setclasspath.sh 在所有代码之前添加 export JAVA_OPTS="-Duser.timezone=GMT" 如果设置了随机启动脚本要修改随机启动脚本 vim /etc/init.d/tomcat export JAVA_OPTS="-Duser.timezone=GMT+8:00" 或 export JAVA_OPTS="-Duser.timezone=GMT+8"
经过以上设置,不管你的程序放到什么地方,每次获取的时间都是北京时间。