MyBatis+MyBatis CDI+PostgreSQL的java时区问题
我在MyBatis有一个奇怪的时区问题
环境:
- mybatis v3。4.5
- mybatis cdi v1。0.1
- Payara应用服务器v4。1.2.173
- PostgreSQL数据库
- JNDI数据源,托管事务设置
首先我将在有日期时间字段的表中插入一条记录。
在Java方面,它是一个java.time.ZonedDateTime
。在数据库中,我对这个字段使用TIMESTAMP WITH TIME ZONE
这是一段相关的代码:
request.setRequestDate = ZonedDateTime.now();
LOGGER.info("xxx-xxx-xxx: " + httpRequest.getRequestDate());
myMapper.save(request);
在日志文件中,我可以看到:
[INFO ] 2017-10-18 18:43:45,501 com..................... - xxx-xxx-xxx: 2017-10-18T18:43:45.493+02:00[Europe/Prague]
[DEBUG] 2017-10-18 18:43:45,608 com...xxxDao.saveRequest - ==> Preparing: INSERT INTO table_a (id, ..., request_date) VALUES (?, ?, ...)
[DEBUG] 2017-10-18 18:43:45,770 com...xxxDao.saveRequest - ==> Parameters: 281(Long), ..., 2017-10-18 16:43:45.493(Timestamp)
[DEBUG] 2017-10-18 18:43:45,783 com...xxxDao.saveRequest - <== Updates: 1
在数据库中,这是结果:
select request_date from table_a
result: 2017-10-18 16:43:45.493
这没关系,因为我在数据库中使用GMT时区
然后我在同一数据源上使用不同的映射器类执行另一个sql插入(POJO也不同),但Java和sql类型是相同的。问题是日期时间值与GMT+2
时区一起插入数据库,而不是GMT
:
[INFO ] 2017-10-18 18:43:46,188 com..................... - yyy-yyy-yyy: 2017-10-18T18:43:46.188+02:00[Europe/Prague]
[DEBUG] 2017-10-18 18:43:46,190 com...yyyDao.saveMetadata - ==> Preparing: INSERT INTO table_b (id,..., uploaded) VALUES (?, ?, ...)
[DEBUG] 2017-10-18 18:43:46,202 com...yyyDao.saveMetadata - ==> Parameters: 561(Long), ..., 2017-10-18 18:43:46.188(Timestamp)
[DEBUG] 2017-10-18 18:43:46,212 com...yyyDao.saveMetadata - <== Updates: 1
如您所见,该值插入的时区错误:
select uploaded from table_b
result: 2017-10-18 18:43:46.188
我的第一个猜测是它来自错误的数据库池设置,因此我在池配置中添加了以下属性:
sessionTimeZone=UTC
但我仍然在UTC中插入第一个日期时间,在UTC+2时区中插入第二个日期时间
我已尝试注销java的详细信息。sql。连接以查看这里到底发生了什么,所以我将SQLSqSession注入到代码中(@InjectSQLSession SqlSession),但当然我得到了一个org.apache.ibatis.session.SqlSessionException: Error: Cannot get connection. No managed session is started..
异常,因为我使用托管事务处理
知道要检查什么吗
----更新1---
因此,我根据建议将所有内容更改为UTC:
- postgresql。形态:时区='UTC'
- 应用服务器:-Duser。时区=UTC
看起来不错,但是
在PostgreSQL控制台中执行查询时,结果如下:
demo=# select EXTRACT(TIMEZONE FROM uploaded), uploaded from image_metadata;
date_part | uploaded
-----------+----------------------------
0 | 2017-10-25 00:24:11.873+00
(1 row)
SqurielSQL中的查询结果相同:
date_part uploaded
0 2017-10-25 02:24:11.873
我的SQL客户端似乎在后台进行了一些棘手的时区对话,但我不确定。我需要检查一下
----更新2---
我不确定问题出在哪里,但在将数据库和JVM时区设置更改为UTC后,我的问题得到了解决
我使用以下SQL查询从数据库中获取存储的日期时间值:
SELECT current_timestamp AT TIME ZONE 'UTC'
# 1 楼答案
我会将一切设置为UTC,使一切变得更简单
如果显示的不是UTC,则将VM参数
-Duser.timezone=UTC
添加到应用程序启动中代码段变量
request
/httpRequest
中存在混乱,可能只是一个输入错误