zoukankan      html  css  js  c++  java
  • JAVA总结---序列化的三种方式

    序列化和反序列化
    序列化:可以将对象转化成一个字节序列,便于存储。
    反序列化:将序列化的字节序列还原
    优点:可以实现对象的"持久性”, 所谓持久性就是指对象的生命周期不取决于程序。

    序列化需要:
    所需类:ObjectInputStream和ObjectOutputStream
    方法: readObject()和writeObject();


    序列化方式一: 实现Serializable接口(隐式序列化)
    通过实现Serializable接口,这种是隐式序列化(不需要手动),这种是最简单的序列化方式,会自动序列化所有非statictransient关键字修饰的成员变量。
     

    1. class Student implements Serializable{
    2. private String name;
    3. private int age;
    4. public static int QQ = 1234;
    5. private transient String address = "CHINA";
    6. Student(String name, int age ){
    7. this.name = name;
    8. this.age = age;
    9. }
    10. public String toString() {
    11. return "name: " + name + " "
    12. +"age: " + age + " "
    13. +"QQ: " + QQ + " "
    14. + "address: " + address;
    15. }
    16. public void SetAge(int age) {
    17. this.age = age;
    18. }
    19. }
    20. public class SerializableDemo {
    21. public static void main(String[] args) throws IOException, ClassNotFoundException {
    22. //创建可序列化对象
    23. System.out.println("原来的对象:");
    24. Student stu = new Student("Ming", 16);
    25. System.out.println(stu);
    26. //创建序列化输出流
    27. ByteArrayOutputStream buff = new ByteArrayOutputStream();
    28. ObjectOutputStream out = new ObjectOutputStream(buff);
    29. //将序列化对象存入缓冲区
    30. out.writeObject(stu);
    31. //修改相关值
    32. Student.QQ = 6666; // 发现打印结果QQ的值被改变
    33. stu.SetAge(18); //发现值没有被改变
    34. //从缓冲区取回被序列化的对象
    35. ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(buff.toByteArray()));
    36. Student newStu = (Student) in.readObject();
    37. System.out.println("序列化后取出的对象:");
    38. System.out.println(newStu);
    39. }
    40. }

    打印结果:
    原来的对象:
    name: Ming
    age: 16
    QQ: 1234
    address: CHINA
    序列化后取出的对象:
    name: Ming
    age: 16
    QQ: 6666
    address: null

    发现address(被transient)和QQ(被static)也没有被序列化,中途修改QQ的值是为了以防读者误会QQ被序列化了。因为序列化可以保存对象的状态,但是QQ的值被改变了,说明没有被序列化。static成员不属于对象实例,可能被别的对象修改没办法序列化,序列化是序列对象。对于address被反序列化后由于没有对应的引用,所以为null。而且Serializable不会调用构造方法。
    PS:细心的可能发现序列化很诱人,可以保存对象的初始信息,在以后可以回到这个初始状态。

    序列化方式二:实现Externalizable接口。(显式序列化)
    Externalizable接口继承自Serializable, 我们在实现该接口时,必须实现writeExternal()和readExternal()方法,而且只能通过手动进行序列化,并且两个方法是自动调用的,因此,这个序列化过程是可控的,可以自己选择哪些部分序列化
     

    1. public class Blip implements Externalizable{
    2. private int i ;
    3. private String s;
    4. public Blip() {}
    5. public Blip(String x, int a) {
    6. System.out.println("Blip (String x, int a)");
    7. s = x;
    8. i = a;
    9. }
    10. public String toString() {
    11. return s+i;
    12. }
    13. @Override
    14. public void writeExternal(ObjectOutput out) throws IOException {
    15. // TODO Auto-generated method stub
    16. System.out.println("Blip.writeExternal");
    17. out.writeObject(s);
    18. out.writeInt(i);
    19. //
    20. }
    21. @Override
    22. public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
    23. // TODO Auto-generated method stub
    24. System.out.println("Blip.readExternal");
    25. s = (String)in.readObject();
    26. i = in.readInt();
    27. }
    28. public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {
    29. System.out.println("Constructing objects");
    30. Blip b = new Blip("A Stirng", 47);
    31. System.out.println(b);
    32. ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("F://Demo//file1.txt"));
    33. System.out.println("保存对象");
    34. o.writeObject(b);
    35. o.close();
    36. //获得对象
    37. System.out.println("获取对象");
    38. ObjectInputStream in = new ObjectInputStream(new FileInputStream("F://Demo//file1.txt"));
    39. System.out.println("Recovering b");
    40. b = (Blip)in.readObject();
    41. System.out.println(b);
    42. }
    43. }

    打印结果为:
    Constructing objects
    Blip (String x, int a)
    A Stirng47
    保存对象
    Blip.writeExternal
    获取对象
    Recovering b
    Blip.readExternal
    A Stirng47

    当注释掉writeExternal和readExternal方法后打印结果为:
    Constructing objects
    Blip (String x, int a)
    A Stirng47
    保存对象
    Blip.writeExternal
    获取对象
    Recovering b
    Blip.readExternal
    null0

    说明:Externalizable类会调用public的构造函数先初始化对象,在调用所保存的内容将对象还原。假如构造方法不是public则会出现运行时错误。

    序列化方式三:实现Serializable接口+添加writeObject()和readObject()方法。(显+隐序列化)

    如果想将方式一和方式二的优点都用到的话,可以采用方式三, 先实现Serializable接口,并且添加writeObject()和readObject()方法。注意这里是添加,不是重写或者覆盖。但是添加的这两个方法必须有相应的格式。

    • 1,方法必须要被private修饰                                ----->才能被调用
    • 2,第一行调用默认的defaultRead/WriteObject(); ----->隐式序列化非static和transient
    • 3,调用read/writeObject()将获得的值赋给相应的值  --->显式序列化
    1. public class SerDemo implements Serializable{
    2. public transient int age = 23;
    3. public String name ;
    4. public SerDemo(){
    5. System.out.println("默认构造器。。。");
    6. }
    7. public SerDemo(String name) {
    8. this.name = name;
    9. }
    10. private void writeObject(ObjectOutputStream stream) throws IOException {
    11. stream.defaultWriteObject();
    12. stream.writeInt(age);
    13. }
    14. private void readObject(ObjectInputStream stream) throws ClassNotFoundException, IOException {
    15. stream.defaultReadObject();
    16. age = stream.readInt();
    17. }
    18. public String toString() {
    19. return "年龄" + age + " " + name;
    20. }
    21. public static void main(String[] args) throws IOException, ClassNotFoundException {
    22. SerDemo stu = new SerDemo("Ming");
    23. ByteArrayOutputStream bout = new ByteArrayOutputStream();
    24. ObjectOutputStream out = new ObjectOutputStream(bout);
    25. out.writeObject(stu);
    26. ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(bout.toByteArray()));
    27. SerDemo stu1 = (SerDemo) in.readObject();
    28. System.out.println(stu1);
    29. }
    30. }

    打印结果为:
    年龄23  Ming
    注释掉stream.writeInt(age)和age= stream.readInt()后:
    年龄0  Ming
    方式三结合了显式和隐式序列化,Ming被正常序列化,由于age被trancient修饰,所以需要显式序列化。

     

    原文地址:https://blog.csdn.net/AHuqihua/article/details/81331316
  • 相关阅读:
    高性能计算发展简史
    软件里有“slave”关键字算不算种族歧视
    LAXCUS集群的松耦合和紧耦合性能对比
    《操作系统》课程笔记(Ch12-大容量存储结构)
    《剑指Offer》部分简单题题解
    ASP.Net Core 5.0 MVC中AOP思想的体现(五种过滤器)并结合项目案例说明过滤器的用法
    扒一扒 剪视频-致力打造最好用的视频创作导航
    C#设计模式02——原型模式的写法
    C#设计模式03——简单工厂的写法
    C#设计模式04——工厂方法的写法
  • 原文地址:https://www.cnblogs.com/jpfss/p/11065362.html
Copyright © 2011-2022 走看看