Почему writeObject выдает исключение java.io.NotSerializableException и как это исправить?

110

У меня есть это исключение, и я не понимаю, почему оно будет выброшено или как мне с этим справиться.

try {
    os.writeObject(element);
} catch (IOException e) {
    e.printStackTrace();
}

Где elementнаходится TransformGroupдругой TransformGroupsэкземпляр класса Atom:

public class Atom extends Group implements Serializable{
    float pozX,pozY;
    Group group= new Group();   
    Color3f blue = new Color3f(new Color(255));
    Color3f black = new Color3f(new Color(0));
    Sphere AtSph=new Sphere();

    public Atom(final float WEIGHT, final int BOUNDS,final float radius,Color3f color)
    {
        AppSetting ap= new AppSetting(color, black);
        AtSph=new Sphere(radius,1,100,ap);
    }
}

Полный журнал ошибок:

java.io.NotSerializableException: javax.media.j3d.TransformGroup
    at java.io.ObjectOutputStream.writeObject0(Unknown Source)
    at java.io.ObjectOutputStream.writeObject(Unknown Source)
    at cls.MolecularBuilder.addAtom(MolecularBuilder.java:511)
    at cls.MolecularBuilder$Console.HidrogenItemActionPerformed(MolecularBuilder.java:897)
    at cls.MolecularBuilder$Console$2.actionPerformed(MolecularBuilder.java:746)
    at javax.swing.AbstractButton.fireActionPerformed(Unknown Source)
    at javax.swing.AbstractButton$Handler.actionPerformed(Unknown Source)
    at javax.swing.DefaultButtonModel.fireActionPerformed(Unknown Source)
    at javax.swing.DefaultButtonModel.setPressed(Unknown Source)
    at javax.swing.AbstractButton.doClick(Unknown Source)
    at javax.swing.plaf.basic.BasicMenuItemUI.doClick(Unknown Source)
    at javax.swing.plaf.basic.BasicMenuItemUI$Handler.mouseReleased(Unknown Source)
    at java.awt.Component.processMouseEvent(Unknown Source)
    at javax.swing.JComponent.processMouseEvent(Unknown Source)
    at java.awt.Component.processEvent(Unknown Source)
    at java.awt.Container.processEvent(Unknown Source)
    at java.awt.Component.dispatchEventImpl(Unknown Source)
    at java.awt.Container.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.LightweightDispatcher.retargetMouseEvent(Unknown Source)
    at java.awt.LightweightDispatcher.processMouseEvent(Unknown Source)
    at java.awt.LightweightDispatcher.dispatchEvent(Unknown Source)
    at java.awt.Container.dispatchEventImpl(Unknown Source)
    at java.awt.Window.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
    at java.awt.EventQueue.access$200(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue.dispatchEvent(Unknown Source)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.run(Unknown Source)

AppSetting (в классе Atom) - это просто настраиваемый класс, расширяющий внешний вид.

Михай Бужанка
источник
4
Похоже, что javax.media.j3d.TransformGroupсам по себе не реализует Serializable
micha
Почему Atomи расширяется, Groupи есть Groupчлен?
Маркиз Лорн,

Ответы:

210

Поля вашего объекта, в свою очередь, имеют свои поля, некоторые из которых не реализуются Serializable. В вашем случае класс-нарушитель TransformGroup. Как это решить?

  • если класс твой, сделай это Serializable
  • если класс сторонний, но в сериализованной форме он вам не нужен, отметьте поле как transient
  • если вам нужны его данные и это сторонние лица, рассмотрите другие средства сериализации, такие как JSON, XML, BSON, MessagePack и т. д., где вы можете сериализовать сторонние объекты без изменения их определений.
Божо
источник
2
Что ж, большое спасибо, это будет довольно сложно сделать, поскольку мои объекты существуют только как часть TransformGroup, и нет никаких переменных, которые их хранят. Мое приложение представляет собой конструктор трехмерных молекул, и все мои атомы и границы просто добавляются в TransformGroup как экземпляры, например (new Atom ()). Проблема не только в том, что мне нужно, чтобы они были записаны в файл, но и в том, что пользователь может захотеть удалить или отредактировать текущие объекты. Думаю, я попробую сериализацию на основе XML, но я новичок в этой концепции, и для меня это пока немного сложно. Спасибо
Михай Буянка
15
Чтобы добавить к этому отличному ответу: если вам нужны его данные и его сторонние лица, вы можете обернуть сторонний класс в свой собственный класс, который реализует Serializableи использует, readObject()а writeObject()также вручную сериализует данные стороннего класса. В некоторых случаях это может быть разумным подходом. stackoverflow.com/a/12963580/1208581
sulai 09
76

java.io.NotSerializableExceptionможет возникнуть при сериализации экземпляра внутреннего класса, потому что:

сериализация такого экземпляра внутреннего класса также приведет к сериализации связанного с ним экземпляра внешнего класса

Сериализация внутренних классов (т. Е. Вложенных классов, которые не являются статическими классами-членами), включая локальные и анонимные классы, настоятельно не рекомендуется.

Ссылка: Сериализуемый интерфейс

То
источник
5
Так было со мной. Это произошло, когда я сократил модульный тест. Надеюсь, что этот ответ сэкономит время кому-то еще.
user489041
У меня было поле private final Set <ClaimsNode> outgoing = new TreeSet <ClaimsNode> (new Comparator <ClaimsNode> () {public int compare (ClaimsNode o1, ClaimsNode o2) {return o1.getNativeIndex () - o2.getNativeIndex (); }});
Виталий Сазанович
1
Удивительно, буквально я был в этом в течение последнего часа. Я начал сомневаться, что даже примитивные целые числа не сериализуемы, а потом мне пришло в голову, что, возможно, здесь что-то действительно не так.
Шивам Похриял
13

Сделайте класс сериализуемым, реализовав интерфейс java.io.Serializable.

  • java.io.Serializable - Интерфейс маркера, в котором нет никаких методов.
  • Назначение интерфейса маркера - сообщить, ObjectOutputStreamчто этот объект является сериализуемым объектом.
Прабхакар
источник
7
Вы можете прочитать вопрос, он уже реализует Serializable. Еще 12+ меня удивляют.
shaILU