Hibernate пытается сериализовать поле, помеченное @Transient - PullRequest
4 голосов
/ 09 декабря 2011

Я пытаюсь сохранить набор из того, что я бы назвал относительно простыми объектами, в базу данных. К сожалению, Hibernate, кажется, игнорирует аннотацию @Transient (или пытается сериализовать, даже с аннотацией) на некоторых из моих классов. Это вызывает неприятную ошибку, когда у меня есть классы, которые нельзя сериализовать.

Если необходимо, я могу сделать сериализуемый класс AxisEventHandler, однако, если я не смог (например, это был последний класс в библиотеке с закрытым исходным кодом), что бы я тогда делал?

Я могу предоставить больше информации о классе GenericEventHandler или любом коде, который я закомментировал для краткости по запросу (но я не думаю, что они особенно важны ... Я могу ошибаться).

Исключение:

org.hibernate.type.SerializationException: could not serialize
    at org.hibernate.util.SerializationHelper.serialize(SerializationHelper.java:139)
    at org.hibernate.util.SerializationHelper.serialize(SerializationHelper.java:164)
    at org.hibernate.util.SerializationHelper.clone(SerializationHelper.java:95)
    at org.hibernate.type.descriptor.java.SerializableTypeDescriptor$SerializableMutabilityPlan.deepCopyNotNull(SerializableTypeDescriptor.java:54)
    at org.hibernate.type.descriptor.java.SerializableTypeDescriptor$SerializableMutabilityPlan.deepCopyNotNull(SerializableTypeDescriptor.java:42)
    at org.hibernate.type.descriptor.java.MutableMutabilityPlan.deepCopy(MutableMutabilityPlan.java:58)
    at org.hibernate.type.AbstractStandardBasicType.deepCopy(AbstractStandardBasicType.java:314)
    at org.hibernate.type.AbstractStandardBasicType.deepCopy(AbstractStandardBasicType.java:310)
    at org.hibernate.type.TypeHelper.deepCopy(TypeHelper.java:68)
    at org.hibernate.event.def.AbstractSaveEventListener.performSaveOrReplicate(AbstractSaveEventListener.java:302)
    at org.hibernate.event.def.AbstractSaveEventListener.performSave(AbstractSaveEventListener.java:203)
    at org.hibernate.event.def.AbstractSaveEventListener.saveWithGeneratedId(AbstractSaveEventListener.java:129)
    at org.hibernate.event.def.DefaultSaveOrUpdateEventListener.saveWithGeneratedOrRequestedId(DefaultSaveOrUpdateEventListener.java:210)
    at org.hibernate.event.def.DefaultSaveEventListener.saveWithGeneratedOrRequestedId(DefaultSaveEventListener.java:56)
    at org.hibernate.event.def.DefaultSaveOrUpdateEventListener.entityIsTransient(DefaultSaveOrUpdateEventListener.java:195)
    at org.hibernate.event.def.DefaultSaveEventListener.performSaveOrUpdate(DefaultSaveEventListener.java:50)
    at org.hibernate.event.def.DefaultSaveOrUpdateEventListener.onSaveOrUpdate(DefaultSaveOrUpdateEventListener.java:93)
    at org.hibernate.impl.SessionImpl.fireSave(SessionImpl.java:713)
    at org.hibernate.impl.SessionImpl.save(SessionImpl.java:701)
    at org.hibernate.impl.SessionImpl.save(SessionImpl.java:697)
    at com.mypandafinance.amino.gui.ccsimple.CCSimpleApp.initApplication(CCSimpleApp.java:55)
    at com.mypandafinance.amino.AminoBoot.main(AminoBoot.java:101)
Caused by: java.io.NotSerializableException: com.mypandafinance.chartcomponent.eventhandlers.AxisEventHandler
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1180)
    at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1528)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1493)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1528)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1493)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1528)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1493)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:346)
    at org.hibernate.util.SerializationHelper.serialize(SerializationHelper.java:135)
    ... 21 more

Класс главной оси:

@Entity
@Inheritance(strategy=InheritanceType.JOINED)
public abstract class Axis implements Serializable {
    private int axisId;

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    public int getAxisId() { return axisId; }
    protected void setAxisId(int id) { this.axisId = id; }

    @Transient
    public abstract long getMax();
    public abstract void setMax(long max);

    @Transient
    public abstract long getMin();
    public abstract void setMin(long min);

    /* ... helper methods here */

    @Transient
    public abstract AxisEventHandler getEventHandler();

    @Transient
    public abstract AxisFlagHandler getFlagHandler();

    @Transient
    public abstract TickMarkFormatter getTickMarkFormatter();
    public abstract void setTickMarkFormatter(TickMarkFormatter formatter);
}

Базовая реализация:

@Entity
@SecondaryTable(name="BasicAxis")
public class BasicAxis extends Axis {
    private long min;
    private long max;

    // Event handler, used for firing off events to all listeners
    private AxisEventHandler eventHandler = new AxisEventHandler();

    // Flag handler, used for storing axis flags
    private SerializableAxisFlagHandler flagHandler;

    private TickMarkFormatter formatter;

    /* ... constructors */

    @Column(name="rangemax")
    @Override public long getMax() { return max; }
    @Override public void setMax(long max) { /* ... */ }

    @Column(name="rangemin")
    @Override public long getMin() { return min; }
    @Override public void setMin(long min) { /* ... */ }

    @Override public void setRange(long min, long max) { /* ... */ }
    @Override public void pan(long amount) { /* ... */ }

    @Transient
    @Override public AxisEventHandler getEventHandler() { /* ... */ }

    @Type(type="com.mypandafinance.chartcomponent.hibernateusertypes.AxisFlagHandlerUserType")
    @Column(name="flags")
    @Override public SerializableAxisFlagHandler getFlagHandler() { /* ... */ }
    protected void setFlagHandler(SerializableAxisFlagHandler flagHandler) { /* ... */ }

    @Column(name="formatter", table="BasicAxis")
    @Override public TickMarkFormatter getTickMarkFormatter() { /* ... */ }
    @Override public void setTickMarkFormatter(TickMarkFormatter formatter) { /* ... */ }

    /* ... helper methods */
}

AxisEventHandler класс:

public class AxisEventHandler extends GenericEventHandler<AxisEventListener> {
    public AxisEventHandler() { super(new AxisEventListener[0]); }
}

Ответы [ 2 ]

4 голосов
/ 09 декабря 2011

Не берите в голову, я, кажется, совершил ошибку, которая, оглядываясь назад, не должна была занять несколько часов, чтобы понять.Проблема была не в hibernate, или в чем-то связанном с Hibernate (кроме того, что Hibernate использовал стандартную сериализацию java).

Мало того, что поля должны быть помечены аннотацией JPA @Transient, но они также должныбыть переходным для интерфейса Serializable (потому что по какой-то причине Hibernate сериализует объекты перед сохранением их?)

Исправлено:

public class BasicAxis extends Axis {
    /* ... */
   private transient AxisEventHandler eventHandler = new AxisEventHandler();
    /* ... */
}
1 голос
/ 30 июня 2013

Это только что случилось со мной, и я тоже потерял часы на этом. Хотя добавление Java-модификатора transient избавило от немедленной ошибки, я продолжал получать все более и более странные ошибки сериализации, пока окончательный Hibernate не пожаловался на то, что не может заклинить двоичное представление моего объекта в поле (потому что это было слишком долго:

Caused by: org.h2.jdbc.JdbcBatchUpdateException: Value too long for column

Реальная проблема? Я случайно облажался с аннотацией JPA. Я сделал это:

@Column(name = "val_foo")
private Foo val;

Вместо этого:

@OneToOne(targetEntity = FooImpl.class, cascade = { PERSIST })
@JoinColumn(name = "val_foo", referencedColumnName = "uuid")
private Foo val;

Надеюсь, это когда-нибудь кому-нибудь поможет ...

...