我很难使用 Hibernate 5.3.x 在 PostgreSQL 中存储与时区无关的timestamp without time zone
值,而不会因不需要的时区转换而损坏值。这是一个众所周知的问题,这里有一个有点规范的问题:如何使用 JPA 和 Hibernate 以 UTC 时区存储日期/时间和时间戳(这谈到使用 UTC,而不是"无时区",但就我的目的而言,它足够接近(。
规范的解决方案是使用 <property name="hibernate.jdbc.time_zone" value="UTC"/>
.
此解决方案对我不起作用。当我查看Hibernate的代码时,我想知道它怎么可能对任何人有用?以下是来源:
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
*/
package org.hibernate.type.descriptor.java;
import java.sql.Timestamp;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import org.hibernate.type.LocalDateTimeType;
import org.hibernate.type.descriptor.WrapperOptions;
/**
* Java type descriptor for the LocalDateTime type.
*
* @author Steve Ebersole
*/
public class LocalDateTimeJavaDescriptor extends AbstractTypeDescriptor<LocalDateTime> {
/**
* Singleton access
*/
public static final LocalDateTimeJavaDescriptor INSTANCE = new LocalDateTimeJavaDescriptor();
@SuppressWarnings("unchecked")
public LocalDateTimeJavaDescriptor() {
super( LocalDateTime.class, ImmutableMutabilityPlan.INSTANCE );
}
@Override
public String toString(LocalDateTime value) {
return LocalDateTimeType.FORMATTER.format( value );
}
@Override
public LocalDateTime fromString(String string) {
return LocalDateTime.from( LocalDateTimeType.FORMATTER.parse( string ) );
}
@Override
@SuppressWarnings("unchecked")
public <X> X unwrap(LocalDateTime value, Class<X> type, WrapperOptions options) {
if ( value == null ) {
return null;
}
if ( LocalDateTime.class.isAssignableFrom( type ) ) {
return (X) value;
}
if ( java.sql.Timestamp.class.isAssignableFrom( type ) ) {
Instant instant = value.atZone( ZoneId.systemDefault() ).toInstant();
return (X) java.sql.Timestamp.from( instant );
}
if ( java.sql.Date.class.isAssignableFrom( type ) ) {
Instant instant = value.atZone( ZoneId.systemDefault() ).toInstant();
return (X) java.sql.Date.from( instant );
}
if ( java.sql.Time.class.isAssignableFrom( type ) ) {
Instant instant = value.atZone( ZoneId.systemDefault() ).toInstant();
return (X) java.sql.Time.from( instant );
}
if ( java.util.Date.class.isAssignableFrom( type ) ) {
Instant instant = value.atZone( ZoneId.systemDefault() ).toInstant();
return (X) java.util.Date.from( instant );
}
if ( Calendar.class.isAssignableFrom( type ) ) {
return (X) GregorianCalendar.from( value.atZone( ZoneId.systemDefault() ) );
}
if ( Long.class.isAssignableFrom( type ) ) {
Instant instant = value.atZone( ZoneId.systemDefault() ).toInstant();
return (X) Long.valueOf( instant.toEpochMilli() );
}
throw unknownUnwrap( type );
}
@Override
public <X> LocalDateTime wrap(X value, WrapperOptions options) {
if ( value == null ) {
return null;
}
if ( LocalDateTime.class.isInstance( value ) ) {
return (LocalDateTime) value;
}
if ( Timestamp.class.isInstance( value ) ) {
final Timestamp ts = (Timestamp) value;
return LocalDateTime.ofInstant( ts.toInstant(), ZoneId.systemDefault() );
}
if ( Long.class.isInstance( value ) ) {
final Instant instant = Instant.ofEpochMilli( (Long) value );
return LocalDateTime.ofInstant( instant, ZoneId.systemDefault() );
}
if ( Calendar.class.isInstance( value ) ) {
final Calendar calendar = (Calendar) value;
return LocalDateTime.ofInstant( calendar.toInstant(), calendar.getTimeZone().toZoneId() );
}
if ( Date.class.isInstance( value ) ) {
final Timestamp ts = (Timestamp) value;
final Instant instant = Instant.ofEpochMilli( ts.getTime() );
return LocalDateTime.ofInstant( instant, ZoneId.systemDefault() );
}
throw unknownWrap( value.getClass() );
}
}
查看ZoneId.systemDefault()
被引用了多少次?它显然没有使用任何属性。我在这里错过了什么?为什么设置此属性可以解决其他人的问题,而不是我?
设置后,Hibernate将调用JDBC PreparedStatement.setTimestamp((方法,该方法采用特定的日历实例。它不是在 LocalDateTimeJavaDescriptor 中完成的。