Java 8 LocalDateTime и Hibernate 4

У меня есть следующий фрагмент описания класса:

... 
@Column(name = "invalidate_token_date")
@Temporal(TemporalType.TIMESTAMP)
private LocalDateTime invalidateTokenDate;
....

Этот код не работает, потому что @Temporal не поддерживает LocalDateTime. Я видел предложение использовать LocalDateTime из Joda-Time, но я использую Java 8.

Пожалуйста, проконсультируйтесь со мной.


P.S.
Вот моя текущая зависимость от JPA:

<dependency>
    <groupId>javax.persistence</groupId>
    <artifactId>persistence-api</artifactId>
    <version>1.0</version>
</dependency>

Ответы

Ответ 1

Так как Hibernate не поддерживает его, вам нужно реализовать пользовательский тип, как показано в этот пример.

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

import java.io.Serializable;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Types;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.util.Date;

public class LocalDateTimeUserType implements EnhancedUserType, Serializable {

    private static final int[] SQL_TYPES = new int[]{Types.TIMESTAMP};

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

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

    @Override
    public boolean equals(Object x, Object y) throws HibernateException {
        if (x == y) {
            return true;
        }
        if (x == null || y == null) {
            return false;
        }
        LocalDateTime dtx = (LocalDateTime) x;
        LocalDateTime dty = (LocalDateTime) y;
        return dtx.equals(dty);
    }

    @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 timestamp = StandardBasicTypes.TIMESTAMP.nullSafeGet(resultSet, names, session, owner);
        if (timestamp == null) {
            return null;
        }
        Date ts = (Date) timestamp;
        Instant instant = Instant.ofEpochMilli(ts.getTime());
        return LocalDateTime.ofInstant(instant, ZoneId.systemDefault());
    }

    @Override
    public void nullSafeSet(PreparedStatement preparedStatement, Object value, int index, SessionImplementor session)
            throws HibernateException, SQLException {
        if (value == null) {
            StandardBasicTypes.TIMESTAMP.nullSafeSet(preparedStatement, null, index, session);
        } else {
            LocalDateTime ldt = ((LocalDateTime) value);
            Instant instant = ldt.atZone(ZoneId.systemDefault()).toInstant();
            Date timestamp = Date.from(instant);
            StandardBasicTypes.TIMESTAMP.nullSafeSet(preparedStatement, timestamp, 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 LocalDateTime.parse(string);
    }

}

Новый тип пользователя можно затем использовать при сопоставлении с аннотацией @Type. Например,

@Type(type="com.hibernate.samples.type.LocalDateTimeUserType")
@Column(name = "invalidate_token_date")
private LocalDateTime invalidateTokenDate;

Аннотация @Type требует полного пути к классу, который реализует интерфейс userType; это factory для создания целевого типа отображаемого столбца.

Вот как сделать то же самое в JPA2.1

Ответ 2

Для любого пользователя Hibernate 5.x существует

<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-java8</artifactId>
    <version>5.0.0.Final</version>
</dependency>

Вам не нужно ничего делать. Просто добавьте зависимость, а типы времени Java 8 должны работать, как и любые другие базовые типы, не требуются аннотации.

private LocalDateTime invalidateTokenDate;

Примечание: это не будет сохраняться до типа timestamp. Тестирование с помощью MySQL, оно сохраняет тип datetime.

Ответ 3

Если вы можете использовать Java EE 7, есть более элегантное решение:

→ Реализуйте это:

@Converter(autoApply = true)
public class LocalDateTimeConverter implements AttributeConverter<LocalDateTime, Date> {

    @Override
    public Date convertToDatabaseColumn(LocalDateTime date) {
        if (date == null){
            return null;
        }
        return date.toDate();
    }

    @Override
    public LocalDateTime convertToEntityAttribute(Date value) {
        if (value == null) {
            return null;
        }
        return LocalDateTime.fromDateFields(value);
    }
} 

→ Используйте вот так:

... 
@Column(name = "invalidate_token_date")
private LocalDateTime invalidateTokenDate;
....

Значение (autoApply = true) означает, что @Converter автоматически используется для преобразования каждого свойства LocalDateTime в ваш объект JPA.

Btw, AttributeConverter довольно хорошо подходит для отображения Enums.

Ответ 4

Я создал простой плагин, чтобы мы могли использовать классы java.time. *. В это время реализуются наиболее часто используемые классы. Посмотрите здесь: https://github.com/garcia-jj/jpa-javatime.

Если вы используете Maven, это артефакт конфигурации:

<dependency>
    <groupId>br.com.otavio</groupId>
    <artifactId>jpa-javatime</artifactId>
    <version>0.2</version>
</dependency>

Более подробная информация о том, как использовать на странице проекта.

Спасибо.