Serializable subclass of non-serializable parent class

前端 未结 2 1035
梦毁少年i
梦毁少年i 2020-12-20 14:52

I am hitting a brickwall with serialization of a subclass of Location in android/java

Location is not serializable. I have a first subclass called FALocation that do

相关标签:
2条回答
  • 2020-12-20 15:11

    Is it absolutely necessary to serialize the Location? maybe you could mark it as transient, and obtain it dynamically after deserializing the object. (Anyway, from the documentation ) :

    Q: If class A does not implement Serializable but a subclass B implements Serializable, will the fields of class A be serialized when B is serialized?

    A: Only the fields of Serializable objects are written out and restored. The object may be restored only if it has a no-arg constructor that will initialize the fields of non-serializable supertypes. If the subclass has access to the state of the superclass it can implement writeObject and readObject to save and restore that state.

    So, if the subclass has access to the fields of its non-serializable superclass(es) it can use the writeObject and readObject protocol to implement serialization. Otherwise, there will be fields that won't be possible to serialize.

    0 讨论(0)
  • 2020-12-20 15:19

    Looks like Location does not have public/protected no-arg constructor. Such a constructor is needed for making it available for serialization in subclass.

    http://download.oracle.com/javase/6/docs/api/java/io/Serializable.html says:

    To allow subtypes of non-serializable classes to be serialized, the subtype may assume responsibility for saving and restoring the state of the supertype's public, protected, and (if accessible) package fields. The subtype may assume this responsibility only if the class it extends has an accessible no-arg constructor to initialize the class's state. It is an error to declare a class Serializable if this is not the case. The error will be detected at runtime.

    And same with the words from Serialization specification:

    A Serializable class must do the following: ... Have access to the no-arg constructor of its first nonserializable superclass

    That would explain why you have problems only in deserialization, because naturally constructor is not called during serialization.

    Small example of failing without accessible constructor:

    public class A {
        public A(String some) {};
        private A() {} //as protected or public everything would work
    }
    
    public class B extends A implements Serializable {
        public B() {
            super("");
        }
        //these doesn't really matter
        //private void writeObject(java.io.ObjectOutputStream out) throws IOException {  }
        //private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException { }
    }
    
    public class BSerializer {
    
        public static void main(String ... args) throws Exception {
            B b = new B();
    
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(baos);
            oos.writeObject(b);
            ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(bais);
            B deserialized = (B) ois.readObject();   //InvalidClassException
        }
    }
    
    0 讨论(0)
提交回复
热议问题