0

このソケットは、同じ名前の古いオブジェクトと新しいオブジェクトを受け取り続けます。ただし、readObject を直接使用すると、致命的な例外がスローされます。古いオブジェクトを無視するか、古いオブジェクト コードを変更することを提案する人もいます。クライアントコードを変更することはできません。下位互換性を処理する必要があり ます。


Socket s = ss.accept();
ObjectInputStream lvStream = new ObjectInputStream(s.getInputStream());
Object lvObject = lvStream.readObject();

// The old class but released ( I cannot change it )
public class Apple extends HashMap<String, String> {
    private static final long serialVersionUID = 1L;
}
// The old class but released ( I cannot change it )
public class Apple extends HashMap<String, String> {
    private static final long serialVersionUID = 2L;
}

// The new class in my local ( 1L or 2L still would not solve the issue easily )
public class Apple extends HashMap<String, String> {
    private static final long serialVersionUID = 2L;
}

readObject() の行で、次の致命的な例外が発生しました。

java.io.InvalidClassException ローカル クラスに互換性がありません: ストリーム classdesc serialVersionUID = 1、ローカル クラス serialVersionUID = 2

この 2 種類の Apple オブジェクト データ (ハッシュマップ) をどのように読み取ることができますか?

ありがとうございました。

4

2 に答える 2

0

ObjectInputStream を拡張する限り、

ストリーム SerialVersionUIDにアクセスして、ローカルの SerialVersionUID と比較できます。


Socket s = ss.accept();
// use the extended ObjectInputStream ( DecompressibleInputStream )
DecompressibleInputStreamlvStream = new DecompressibleInputStream(s.getInputStream());
Object lvObject = lvStream.readObject();

import java.io.IOException;
import java.io.InputStream;
import java.io.InvalidClassException;
import java.io.ObjectInputStream;
import java.io.ObjectStreamClass;

public class DecompressibleInputStream extends ObjectInputStream {

    public DecompressibleInputStream(InputStream in) throws IOException {
        super(in);
    }

    protected ObjectStreamClass readClassDescriptor() throws IOException, ClassNotFoundException {
        ObjectStreamClass resultClassDescriptor = super.readClassDescriptor();
        Class<?> localClass;
        try {
            localClass = Class.forName(resultClassDescriptor.getName()); 
        } catch (ClassNotFoundException e) {
            System.err.println("No local class for " + resultClassDescriptor.getName());
            return resultClassDescriptor;
        }
        ObjectStreamClass localClassDescriptor = ObjectStreamClass.lookup(localClass);
        if (localClassDescriptor != null) {
            final long localSUID = localClassDescriptor.getSerialVersionUID();
            final long streamSUID = resultClassDescriptor.getSerialVersionUID();
            if (streamSUID != localSUID) {
                System.err.println("Potentially Fatal Deserialization Operation.");
                resultClassDescriptor = localClassDescriptor;
            }
        }
        return resultClassDescriptor;
    }
}

それが役に立てば幸い

(この投稿のクレジット)

于 2016-12-06T02:08:23.413 に答える