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>です。匿名クラスを実装するために複数のインターフェイスを指定する方法はないと思います(つまり、2 つを組み合わせるために追加のインターフェイスを作成する必要があります) が、間違っている可能性があります。

これで、名前付き内部クラスが を実装Serializableし、その唯一のフィールドの型も実装されるようになり、すべてが機能します。

于 2013-03-02T17:39:38.510 に答える