2009-10-14 16 views
6

Stworzyłem UserType (patrz poniżej) do obsługi sytuacji w naszej bazie danych MySQL, w której zapisujemy daty zerowe jako 0000-00-00 00:00:00.Hibernate niestandardowy UserType nie działa

Kiedy próbuję utrzymać moją jednostkę z wartością null dla dispDT (patrz poniżej), generuje ten wyjątek: "javax.persistence.PersistenceException: org.hibernate.PropertyValueException: właściwość not-null odwołuje się do wartości pustej lub przejściowej: myEntity .dispDt "

Ustawiając punkt przerwania w każdej metodzie w MySQLTimeStampUserType widzę, że wywołuje metodę deepCopy i nigdy nie wywołuje metody nullSafeSet. Sądziłem, że celem metody nuyllSafeSet było umożliwienie mi manipulowania wartością przed jej utrwaleniem. Co ja robię źle?

Entity Adnotacje

@Basic(optional = false) 
@Column(name = "disp_dt") 
@Type(type = "mypackage.MySQLTimeStampUserType") 
// @Temporal(TemporalType.TIMESTAMP) 
private Date dispDt; 

klasa Typ użytkownika

public class MySQLTimeStampUserType implements UserType { 

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

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

public Class returnedClass() { 
    return Date.class; 
} 

public boolean equals(Object x, Object y) throws HibernateException { 
    if (x == y) { 
     return true; 
    } else if (x == null || y == null) { 
     return false; 
    } else { 
     return x.equals(y); 
    } 

} 

public int hashCode(Object arg0) throws HibernateException { 
    throw new UnsupportedOperationException("Not supported yet."); 
} 

public Object nullSafeGet(ResultSet resultSet, String[] names, Object owner) throws HibernateException, SQLException { 
    // if the date is 0000-00-00 00:00:00 return null, else return the timestamp 
    Date result = null; 
    if (!resultSet.wasNull()) { 
     if (!resultSet.getString(names[0]).equals("0000-00-00 00:00:00")) { 
      result = resultSet.getDate(names[0]); 
     } 
    } 
    return result; 
} 

public void nullSafeSet(PreparedStatement statement, Object value, int index) throws HibernateException, SQLException { 
    // if the date is null set the value to "0000-00-00 00:00:00" else save the timestamp 
    if (value == null) { 
     statement.setString(index, "0000-00-00 00:00:00"); 
    } else { 
     statement.setTimestamp(index,(Timestamp) value); 
    } 

} 

public Object deepCopy(Object value) throws HibernateException { 
    return value; 
} 

public boolean isMutable() { 
    return false; 
} 

public Serializable disassemble(Object value) throws HibernateException { 
    throw new UnsupportedOperationException("Not supported yet."); 
} 

public Object assemble(Serializable cached, Object owner) throws HibernateException { 
    throw new UnsupportedOperationException("Not supported yet."); 
} 

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

Odpowiedz

4

Twój problem nie jest ze swojej UserType - to z faktu, że już zadeklarowanej swoją nieruchomość jako nie-zerowy (używając @Basic opcjonalnego = "false"), a mimo to ustawiasz wartość null.

Powiedziałbym, że będę ostrożny, zwracając pierwotną wartość w metodach deepCopy/assemble/disassembly. java.util.Date jest zmienny i możesz poprosić o kłopoty.

+0

Baza danych nie zezwala na wartość pustą dla tej kolumny. Zamiast tego wypełniają go "0000-00-00 00:00:00". Myślałem, że celem UserType było umożliwienie mi przekonwertowania wartości null na "0000-00-00 00:00:00". – Preston

+2

Nie mówię o bazie danych. Ustawiasz tę właściwość na wartość NULL w aplikacji; Hibernacja mierzy się z tym, ponieważ jest mapowany jako opcjonalny = fałsz. Usuń tę deklarację i będziesz gotowy do pracy - baza danych nie uzyska NULL w tej kolumnie, ponieważ twój typ użytkownika zastąpi ją zerami. – ChssPly76

Powiązane problemy