Java无法使用Comparators序列化包含TreeMaps的对象

时间:2014-01-07 18:19:50

标签: java exception serialization treemap notserializableexception

我作为一项任务(来自Uni的OOP课程)是一个非常大的项目:学校注册,学生可以看到他们的成绩,教师可以增加成绩等。

“base”类是一个包含所有使用的类(Java)的单例,例如用户数组,类(如学校类)和将classess和教师关联到课程的TreeMap。

我想序列化这个基类(Central),以便保存修改后的数据。问题是我得到了这个例外

java.io.NotSerializableException: liceu.Central$1
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1183)
at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1547)
at java.io.ObjectOutputStream.defaultWriteObject(ObjectOutputStream.java:440)
at java.util.TreeMap.writeObject(TreeMap.java:2265)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:606)
at java.io.ObjectStreamClass.invokeWriteObject(ObjectStreamClass.java:988)
at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1495)
at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1431)
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1177)
at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1547)
at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1508)
at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1431)
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1177)
at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:347)
at liceu.Main.main(Main.java:31)

我的所有类都实现了Serializable,它们没有瞬态或静态字段(单例除外,它具有实例变量和getInstance方法作为静态)。

因为要发布相当多的代码(并且我会冒着通过在提交之前发布它来使我的作业无效),我试图通过尝试隔离错误来进行概念验证。

public class Central implements Serializable
{
    private ArrayList <User> users;
    private ArrayList <Class> classess;
    private TreeMap <Course, TreeMap <Class, Professor>> reunite;
    private static Central instance = null;

    private Central()
    {
        users = new ArrayList<>();
        classess = new ArrayList<>();
        reunite = new TreeMap<>(new Comparator<Student>(){
            @Override
            public int compare(Student e1, Student e2)
            {
                return e1.getName().compareTo(e2.getName());
            }
        });
    }
}

如果我只保留前2个ArrayLists,则序列化过程有效。 问题在于TreeMap。

TreeMap类是否可序列化? (一般来说) 是因为匿名比较者?

这是序列化的主类

public class Main
{
    public static void main(String args[])
    {
        Central cent = Central.getInstance();
        FileOutputStream fos;
        ObjectOutputStream oos;

        cent.addUser(new Student(3,"id","pass","name","surname"));
        cent.addUser(new Student(3,"id2","pass","name","surname"));
        cent.addUser(new Student(3,"id1","pass","name","surname"));
        try
        {
            fos = new FileOutputStream("save.txt");
            oos = new ObjectOutputStream(fos);
            oos.writeObject(cent);
        }

        catch(IOException e)
        {
            e.printStackTrace();
        }
    }
}

3 个答案:

答案 0 :(得分:12)

TreeMap包含对比较键的比较器的引用。比较器是一个不可序列化的匿名类的实例。所以你得到了这个例外。

将匿名比较器重构为顶层或命名的内部类,它也实现了Serializable。

答案 1 :(得分:4)

强制ComparatorSerializable的另一种方法是将其转换为带有额外绑定强制转换的lambda:

TreeMap<Integer, String> sortedMap = new TreeMap<>(
    (Comparator<Integer> & Serializable) (o1, o2) -> {
        return o1.compareTo(o2);
    }
);

以下是解释: Java lambda expressions, casting, and Comparators

答案 2 :(得分:2)

向Comparator添加“implements Serializable”,它应该可以解决问题。