java – 从Callable返回的本地类的序列化
作者:互联网
这是一个脑力激荡者.
我知道这个实际代码在很多层面都很糟糕.我的问题不是如何做到这一点(我知道静态初始化块),但为什么这不起作用,为了我理解Java序列化的好处.
为什么这样做
import java.io.*;
import java.util.*;
class Main {
static Comparator<String> COMPARE_STRING_LENGTH;
static {
class CompareStringReverse implements Comparator<String>, Serializable {
public int compare(String o1, String o2) {
return o1.length() - o2.length();
}
};
COMPARE_STRING_LENGTH = new CompareStringReverse();
}
public static void main(String[] args) throws IOException {
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("test.ser"));
out.writeObject(new TreeSet<String>(COMPARE_STRING_LENGTH));
out.close();
}
}
而这个
import java.io.*;
import java.util.*;
import java.util.concurrent.Callable;
class Main {
static Comparator<String> COMPARE_STRING_LENGTH = new Callable<Comparator<String>>() {
public Comparator<String> call() {
class CompareStringReverse implements Comparator<String>, Serializable {
public int compare(String o1, String o2) {
return o1.length() - o2.length();
}
};
return new CompareStringReverse();
}
}.call();
public static void main(String[] args) throws IOException {
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("test.ser"));
out.writeObject(new TreeSet<String>(COMPARE_STRING_LENGTH));
out.close();
}
}
产量
Exception in thread "main" java.io.NotSerializableException: Main$1
解决方法:
那么“为什么”有点微妙.即使您的命名类(CompareStringReverse)实现了Serializable,它也嵌套在匿名类中,而不是.因此它是一个内部类,并且隐含引用了封闭匿名类的实例.例如,如果您运行:
javap -c Main$1$1CompareStringReverse
你会看到这个领域:
final Main$1 this$0;
这就是未能序列化的内容.您仍然可以轻松修复它:
interface SerializableCallable<T> extends Serializable, Callable<T> {}
...
static Comparator<String> COMPARE_STRING_LENGTH =
new SerializableCallable<Comparator<String>>() {
...
};
唯一显着的区别是匿名类现在实现Serializable以及Callable< T>.我认为没有任何方法可以为匿名类指定多个接口来实现(即你必须创建额外的接口来组合这两个)但我可能是错的.
所以现在命名的内部类实现了Serializable,其唯一字段的类型也是如此,所以一切正常.
标签:java,exception,serialization,anonymous-class,callable 来源: https://codeday.me/bug/20190529/1179386.html