美文网首页
Java基础 (12) 序列化与IO

Java基础 (12) 序列化与IO

作者: perry_Fan | 来源:发表于2019-02-18 19:38 被阅读0次

    1)序列化的方式
    2)为什么要序列化
    3)Serializable和Parcelable的区别?
    4)如何将一个Java对象序列化到文件里?
    5)Android为什么引入Parcelable?
    6)有没有尝试简化Parcelable的使用?

    一. 概述

    Java 提供了一种对象序列化的机制,该机制中,一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中数据的类型。

    将序列化对象写入文件之后,可以从文件中读取出来,并且对它进行反序列化,也就是说,对象的类型信息、对象的数据,还有对象中的数据类型可以用来在内存中新建对象。

    整个过程都是 Java 虚拟机(JVM)独立的,也就是说,在一个平台上序列化的对象可以在另一个完全不同的平台上反序列化该对象。

    类 ObjectInputStream 和 ObjectOutputStream 是高层次的数据流,它们包含反序列化和序列化对象的方法。
    ObjectOutputStream 类包含很多写方法来写各种数据类型,但是一个特别的方法例外:

    public final void writeObject(Object x) throws IOException
    

    上面的方法序列化一个对象,并将它发送到输出流。相似的 ObjectInputStream 类包含如下反序列化一个对象的方法:

    public final Object readObject() throws IOException, 
                                   ClassNotFoundException
    

    该方法从流中取出下一个对象,并将对象反序列化。它的返回值为Object,因此,你需要将它转换成合适的数据类型。

    为了演示序列化在Java中是怎样工作的,我将使用之前教程中提到的Employee类,假设我们定义了如下的Employee类,该类实现了Serializable 接口。

    public class Employee implements java.io.Serializable
    {
       public String name;
       public String address;
       public transient int SSN;
       public int number;
       public void mailCheck()
       {
          System.out.println("Mailing a check to " + name
                               + " " + address);
       }
    }
    

    请注意,一个类的对象要想序列化成功,必须满足两个条件:

    • 该类必须实现 java.io.Serializable 对象。
    • 该类的所有属性必须是可序列化的。如果有一个属性不是可序列化的,则该属性必须注明是短暂的。
    • 如果你想知道一个 Java 标准类是否是可序列化的,请查看该类的文档。检验一个类的实例是否能序列化十分简单, 只需要查看该类有没有实现 java.io.Serializable接口。
    序列化对象

    ObjectOutputStream 类用来序列化一个对象,如下的 SerializeDemo 例子实例化了一个 Employee 对象,并将该对象序列化到一个文件中。

    import java.io.*;
     
    public class SerializeDemo
    {
       public static void main(String [] args)
       {
          Employee e = new Employee();
          e.name = "Reyan Ali";
          e.address = "Phokka Kuan, Ambehta Peer";
          e.SSN = 11122333;
          e.number = 101;
          try
          {
             FileOutputStream fileOut =
             new FileOutputStream("/tmp/employee.ser");
             ObjectOutputStream out = new ObjectOutputStream(fileOut);
             out.writeObject(e);
             out.close();
             fileOut.close();
             System.out.printf("Serialized data is saved in /tmp/employee.ser");
          }catch(IOException i)
          {
              i.printStackTrace();
          }
       }
    }
    
    反序列化对象
    import java.io.*;
     
    public class DeserializeDemo
    {
       public static void main(String [] args)
       {
          Employee e = null;
          try
          {
             FileInputStream fileIn = new FileInputStream("/tmp/employee.ser");
             ObjectInputStream in = new ObjectInputStream(fileIn);
             e = (Employee) in.readObject();
             in.close();
             fileIn.close();
          }catch(IOException i)
          {
             i.printStackTrace();
             return;
          }catch(ClassNotFoundException c)
          {
             System.out.println("Employee class not found");
             c.printStackTrace();
             return;
          }
          System.out.println("Deserialized Employee...");
          System.out.println("Name: " + e.name);
          System.out.println("Address: " + e.address);
          System.out.println("SSN: " + e.SSN);
          System.out.println("Number: " + e.number);
        }
    }
    

    为什么要序列化呢?

    • 永久性保存对象,保存对象的字节序列到本地文件中;
    • 通过序列化对象在网络中传递对象;
    • 通过序列化在进程间传递对象。
    二. Parcelable 与Serializable 的区别
    1. 在使用内存的时候,Parcelable 类比Serializable性能高,所以推荐使用Parcelable类。
    2. Serializable在序列化的时候会产生大量的临时变量,从而引起频繁的GC。
    3. Parcelable不能使用在要将数据存储在磁盘上的情况,因为Parcelable不能很好的保证数据的持续性在外界有变化的情况下。尽管Serializable效率低点,也不提倡用,但在这种情况下,还是建议你用Serializable 。

    引用链接:http://www.runoob.com/java/java-serialization.html

    三. BIO、NIO、AIO



    NIO (New IO)
    Channel 和缓冲区buffer进行操作,数据总是从通道读取到缓冲区中,或者从缓冲区写入通道也类似
    non-blocking IO 非阻塞IO
    例如:当线程从通道读取数据到缓冲区时,线程还是进行其他事情。当数据被写入到缓冲区时,线程可以继续处理它。
    Selectors 选择器
    用于监听多个通道的事件(连接打开,数据到达)单个线程可以监听多个数据通道。









    相关文章

      网友评论

          本文标题:Java基础 (12) 序列化与IO

          本文链接:https://www.haomeiwen.com/subject/wchyeqtx.html