Grails(Hibernate)将java.time.ZoneId映射到数据库

时间:2016-10-24 08:47:55

标签: hibernate jpa grails grails-3.0 hibernate-5.x

有没有办法在Hibernate 5.1.1中支持java.time.ZoneId持久映射到字符串。它现在以二进制形式保存ZoneId。

我刚刚升级到拥有Hibernate 5.1.1的Grails 3.2.1。例如,保存java.time.Instant可以正常工作,但java.time.ZoneId只能以二进制形式存储。

我认为Hibernate没有支持。那么如何编写自己的映射代码呢?我尝试使用Jadira Framework,但由于启动grails应用程序时存在一些冲突(例外),因此无法实现。

3 个答案:

答案 0 :(得分:4)

您可以使用JPA 2.1定义的自定义属性转换器。像这样声明转换器类:

@Converter
public static class ZoneIdConverter implements AttributeConverter<ZoneId, String> {

    @Override
    public String convertToDatabaseColumn(ZoneId attribute) {
        return attribute.getId();
    }

    @Override
    public ZoneId convertToEntityAttribute(String dbData) {
        return ZoneId.of( dbData );
    }
}

然后从类型ZoneId的实体属性中引用它:

@Convert(converter = ZoneIdConverter.class)
private ZoneId zoneId;

持久化/加载zoneId属性时,将自动调用转换器。

答案 1 :(得分:0)

所以我终于找到了一种很好的方法来实现自定义的hibernate用户类型。要将java.time.ZoneId作为varchar实现持久化以下用户类型类:

import org.hibernate.HibernateException
import org.hibernate.engine.spi.SessionImplementor
import org.hibernate.type.StandardBasicTypes
import org.hibernate.usertype.EnhancedUserType

import java.sql.PreparedStatement
import java.sql.ResultSet
import java.sql.SQLException
import java.sql.Types
import java.time.ZoneId

/**
 * A type that maps between {@link java.sql.Types#VARCHAR} and {@link ZoneId}.
 */
class ZoneIdUserType implements EnhancedUserType, Serializable {

    private static final int[] SQL_TYPES = [Types.VARCHAR]

    @Override
    public int[] sqlTypes() {
        return SQL_TYPES
    }

    @Override
    public Class returnedClass() {
        return ZoneId.class
    }

    @Override
    public boolean equals(Object x, Object y) throws HibernateException {
        if (x == y) {
            return true
        }
        if (x == null || y == null) {
            return false
        }
        ZoneId zx = (ZoneId) x
        ZoneId zy = (ZoneId) y
        return zx.equals(zy)
    }

    @Override
    public int hashCode(Object object) throws HibernateException {
        return object.hashCode()
    }

    @Override
    public Object nullSafeGet(ResultSet resultSet, String[] names, SessionImplementor session, Object owner)
        throws HibernateException, SQLException {
        Object zoneId = StandardBasicTypes.STRING.nullSafeGet(resultSet, names, session, owner)
        if (zoneId == null) {
            return null
        }
        return ZoneId.of(zoneId)
    }

    @Override
    public void nullSafeSet(PreparedStatement preparedStatement, Object value, int index, SessionImplementor session)
        throws HibernateException, SQLException {
        if (value == null) {
            StandardBasicTypes.STRING.nullSafeSet(preparedStatement, null, index, session)
        } else {
            def zoneId = (ZoneId) value
            StandardBasicTypes.STRING.nullSafeSet(preparedStatement, zoneId.getId(), index, session)
        }
    }

    @Override
    public Object deepCopy(Object value) throws HibernateException {
        return value
    }

    @Override
    public boolean isMutable() {
        return false
    }

    @Override
    public Serializable disassemble(Object value) throws HibernateException {
        return (Serializable) value
    }

    @Override
    public Object assemble(Serializable cached, Object value) throws HibernateException {
        return cached
    }

    @Override
    public Object replace(Object original, Object target, Object owner) throws HibernateException {
        return original
    }

    @Override
    public String objectToSQLString(Object object) {
        throw new UnsupportedOperationException()
    }

    @Override
    public String toXMLString(Object object) {
        return object.toString()
    }

    @Override
    public Object fromXMLString(String string) {
        return ZoneId.of(string)
    }
}

然后,您需要在Grails应用的conf/application.groovy中注册自定义用户类型:

grails.gorm.default.mapping = {
    'user-type'(type: ZoneIdUserType, class: ZoneId)
}

您可以在域类中使用java.time.ZoneId:

import java.time.ZoneId

class MyDomain {
    ZoneId zoneId
}

请参阅:

  1. http://docs.grails.org/latest/ref/Database%20Mapping/Usage.html
  2. http://blog.progs.be/550/java-time-hibernate

答案 2 :(得分:0)

您可以使用Hibernate types库,然后编写

@Column
private ZoneId zoneId;

在您的实体类中。您必须使用以下注释标记实体类:

@TypeDef(typeClass = ZoneIdType.class, defaultForType = ZoneId.class)