4

这是一个脑筋急转弯。

我知道这个实际的代码在很多层面上都是可怕的。我的问题不是如何做到这一点(我知道静态初始化块),而是为什么这不起作用,以利于我对 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
4

1 回答 1

7

好吧,“为什么”有点微妙。即使您的命名类 ( 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了,它的唯一字段的类型也是如此,所以一切正常。

于 2013-03-02T17:39:38.510 回答