Java IO&NIO

作者: Tycc | 来源:发表于2018-03-01 21:29 被阅读7次

    Java 的 I/O 操作类在包 java.io 下,大概有将近 80 个类,但是这些类大概可以分成四组,分别是:

    • 基于字节操作的 I/O 接口:InputStream 和 OutputStream
    • 基于字符操作的 I/O 接口:Writer 和 Reader
    • 基于磁盘操作的 I/O 接口:File
    • 基于网络操作的 I/O 接口:Socket

    几种不同的InputStream:

    • FileInputStream把一个文件作为InputStream,实现对文件的读取操作
    • ByteArrayInputStream:把内存中的一个缓冲区作为InputStream使用
    • StringBufferInputStream:把一个String对象作为InputStream
    • PipedInputStream:实现了pipe的概念,主要在线程中使用
    • SequenceInputStream:把多个InputStream合并为一个InputStream

    几种不同的OutputStream:

    • ByteArrayOutputStream:把信息存入内存中的一个缓冲区中
    • FileOutputStream:把信息存入文件中
    • PipedOutputStream:实现了pipe的概念,主要在线程中使用
    • SequenceOutputStream:把多个OutStream合并为一个OutStream

    BufferedWriter 和 BufferedReader 为带有默认缓冲的字符输出输入流,因有缓冲区所以效率比没有缓冲区的很高。

    类的序列化

    序列化 (Serialization)是将对象的状态信息转换为可以存储或传输的格式的过程。 反序列化 (Deserialization)是通过从存储或者网络读取对象的状态,重新创建该对象。序列化广泛应用在远程调用(RPC)或者数据存取。
    Serializable接口,是一个空接口;如果一个类实现了Serializable接口,那么就代表这个类是自动支持序列化和反序列化的。

    //比如如果 employee实现了Serializable接口,然后把类储存到二进制文件中或从二进制文件中提取
        ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("employee.dat"));
             out.writeObject(staff);
             out.close();
    
             ObjectInputStream in = new ObjectInputStream(new FileInputStream("employee.dat"));
             Employee[] newStaff = (Employee[]) in.readObject();
             in.close();
    

    ps:

    1. transient关键字防止字段序列化。在被反序列化后,transient 变量的值被设为初始值,如 int 型的是 0,对象型的是 null。
      private transient int age;
    2. 序列化 ID 决定虚拟机是否允许反序列化。
    public class A implements Serializable { 
     
        private static final long serialVersionUID = 2L; 
      
      ...
    }
    
    1. 为了防止数据序列化,还可以将不需要被序列化的字段抽取出来放到父类中,子类实现 Serializable 接口,父类不实现,根据父类序列化规则,父类的字段数据将不被序列化。

    2. 序列化并不保存静态变量。

    3. 保存多个相同对象时,第一个对象之后保存的是引用。

    Cloneable 的用途

    Cloneable和Serializable一样都是标记型接口,它们内部都没有方法和属性。
    实现该接口表示能使用Object.clone()方法。
    深拷贝还需要重写(override)Object类的clone()方法。

    Socket

    • 创建服务器ServerSocket ss = new ServerSocket(8888) 8888是端口号。
      接受请求 Socket s = ss.accept();
    • 客户端建立连接Socket s = new Socket("127.0.0.1",8888);

    NIO

    NIO 弥补了原来的 I/O 的不足,是非阻塞模式,使一个线程从某通道发送请求读取数据,但是它仅能得到目前可用的数据,如果目前没有数据可用时,就什么都不会获取,而不会保持线程阻塞。所以直至数据变的可以读取之前,该线程可以继续做其他的事情。数据必须先读入缓冲区再处理。

    通道缓冲区是 NIO 中的核心对象,几乎在每一个 I/O 操作中都要使用它们。
    通道是对原 I/O 包中的流的模拟。到任何目的地(或来自任何地方)的所有数据都必须通过一个 Channel 对象。一个 Buffer 实质上是一个容器对象。发送给一个通道的所有对象都必须首先放到缓冲区中;同样地,从通道中读取的任何数据都要读到缓冲区中。

    • 从文件中读取
    //第一步是获取通道。从 FileInputStream 获取通道:
    FileInputStream fin = new FileInputStream( "readandshow.txt" );
    FileChannel fc = fin.getChannel();
    
    //下一步是创建缓冲区:
    ByteBuffer buffer = ByteBuffer.allocate( 1024 );
    
    //最后,需要将数据从通道读到缓冲区中,如下所示:
    fc.read( buffer );
    
    • 写入文件
    //在 NIO 中写入文件类似于从文件中读取。首先从 FileOutputStream 获取一个通道:
    FileOutputStream fout = new FileOutputStream( "writesomebytes.txt" );
    FileChannel fc = fout.getChannel();
    
    //下一步是创建一个缓冲区并在其中放入一些数据 
    //在这里,数据将从一个名为 message 的数组中取出
    //这个数组包含字符串 "Some bytes" 的 ASCII 字节。
    
    ByteBuffer buffer = ByteBuffer.allocate( 1024 );
     
    for (int i=0; i<message.length; ++i) {
         buffer.put( message[i] );
    }
    
    //flip() 方法让缓冲区可以将新读入的数据写入另一个通道。
    buffer.flip();
    
    //最后一步是写入缓冲区中:
    fc.write( buffer );
    
    、、clear() 方法重设缓冲区,使它可以接受读入的数据
    

    参考:

    推荐阅读

    相关文章

      网友评论

        本文标题:Java IO&NIO

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