我作为一项任务(来自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();
}
}
}
答案 0 :(得分:12)
TreeMap包含对比较键的比较器的引用。比较器是一个不可序列化的匿名类的实例。所以你得到了这个例外。
将匿名比较器重构为顶层或命名的内部类,它也实现了Serializable。
答案 1 :(得分:4)
强制Comparator
为Serializable
的另一种方法是将其转换为带有额外绑定强制转换的lambda:
TreeMap<Integer, String> sortedMap = new TreeMap<>(
(Comparator<Integer> & Serializable) (o1, o2) -> {
return o1.compareTo(o2);
}
);
答案 2 :(得分:2)
向Comparator添加“implements Serializable”,它应该可以解决问题。