为什么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它本身未实现可序列化
micha 2012年

为什么Atom既要扩展Group又要有Group成员?
罗恩侯爵

Answers:


210

对象的字段又包含其字段,其中某些字段未实现Serializable。就您而言,令人反感的课程是TransformGroup。怎么解决呢?

  • 如果班级是你的,那就上 Serializable
  • 如果班级是第3方,但您不需要序列化形式,则将该字段标记为 transient
  • 如果您需要其数据并且是第三方,请考虑其他序列化方式,例如JSON,XML,BSON,MessagePack等,您可以在不修改其定义的情况下序列化第三方对象。

2
好吧,非常感谢,这将是一件很难的事,因为我的对象仅作为TransformGroup的一部分存在,并且没有任何存储它们的变量。我的应用程序是3d分子构建器,我的所有原子和边界都作为实例添加到了TransformGroup中,例如(新的Atom())。问题不仅在于我需要将它们写入文件,而且用户可能要删除或编辑当前对象。我想我会尝试一些基于XML的序列化,但是我对这个概念还很陌生,对我来说还是有点困难。谢谢
Mihai Bujanca

15
要补充一个很好的答案:如果您需要它的数据并且是第三方,则可能需要将第三方类包装到您自己的类中,该类实现Serializable并使用readObject()writeObject()手动序列化第三方类的数据。在某些情况下,这可能是一种合理的方法。stackoverflow.com/a/12963580/1208581
苏莱2014年

76

java.io.NotSerializableException序列化内部类实例时可能会发生,因为:

序列化这样的一个内部类实例也将导致其相关的外部类实例的序列化

强烈建议不要对内部类(即不是静态成员类的嵌套类)进行序列化,包括本地和匿名类

参考:可序列化的接口


5
对我来说就是这种情况。我在单元测试中捷径时就发生了这种情况。希望这个答案可以节省一些时间。
user489041

我有一个字段私有的最终Set <ClaimsNode>外出= new TreeSet <ClaimsNode>(new Comparator <ClaimsNode>(){public int compare(ClaimsNode o1,ClaimsNode o2){return o1.getNativeIndex()-o2.getNativeIndex(); }});
Vitaly Sazanovich,

1
从字面上看,我真是太过了一个小时。我开始怀疑,即使是原始int也无法序列化,然后我才想到,这里可能真的有问题。
Shivam Pokhriyal

13

通过实现接口使类可序列化java.io.Serializable

  • java.io.Serializable -标记接口,其中没有任何方法。
  • 标记接口的目的-告知ObjectOutputStream此对象是可序列化的对象。

7
您可以阅读问题,它已经实现了Serializable。还是12岁以上让我惊讶。
shaILU
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.