JAVA NIO
NIO属于多路复用IO模型(详见五种IO模型)。
在Java1.4之前的IO系统中,提供的都是面向流的IO,系统一次一个字节地处理数据,一个输入流产生一个字节的数据,一个输出流消费一个字节的数据,因此面向流的I/O速度非常慢。Java 1.4中推出了NIO,这是一个面向块(缓冲区)的IO系统,系统以块的方式处理,每一个操作在一步中产生或者消费一个数据库,大大提高了处理速度。
NIO三大核心:Buffer(缓冲区)、Channel(通道)、Selector(选择器)。
与传统IO(基于字节流和字符流进行操作)不同,NIO基于Channel(通道)和Buffer(缓冲区)进行操作,数据总是从通道读取到缓冲区中,或者从缓冲区写入到通道中。Selector(选择器)用于监听多个通道的事件(比如:连接打开,数据到达)。因此,单个线程可以监听多个数据通道。
1 Channel 通道
通道是一个对象,通过它可以读取和写入数据。
Channel 和 IO 中的 Stream(流)类似。只不过 Stream 是单向的,(如InputStream/OutputStream)。而 Channel 是双向的,既可以用来进行读操作,也可以用来进行写操作。
NIO 中的 Channel 的主要实现有:
- FileChannel
- DatagramChannel
- SocketChannel
- ServerSocketChannel
一般来说不会将字节直接写入通道中,而是将数据写入包含一个或者多个字节的缓冲区;同样,一般也不会直接从通道中读取字节,而是将数据从通道读入缓冲区,再从缓冲区获取这个字节。
2 Buffer 缓冲区
缓冲区实际上是一个容器对象,更直接的说,是一个连续数组。
Channel 提供从文件、网络读取数据的渠道,但是读取或写入的数据都必须经由 Buffer。
客户端发送数据时,必须先将数据存入 Buffer 中,然后将 Buffer 中的内容写入通道。服务端这边接收数据必须通过 Channel 将数据读入到 Buffer 中,然后再从 Buffer 中取出数据来处理。
在 NIO 中,所有的缓冲区类型都继承于抽象类Buffer。对于Java中的基本类型,基本都有一个具体Buffer类型与之相对应:ByteBuffer、IntBuffer、 CharBuffer、 LongBuffer、 DoubleBuffer、FloatBuffer、ShortBuffer等。
3 Selector 选择器
Selector 类是 NIO 的核心类,Selector 能够检测多个注册的通道上是否有事件发生,如果有事件发生,便获取事件然后针对每个事件进行相应的响应处理。这样一来,只是用一个单线程就可以管理多个通道,也就是管理多个连接。这样使得只有在连接真正有读写事件发生时,才会调用函数来进行读写,就大大地减少了系统开销,并且不必为每个连接都创建一个线程,不用去维护多个线程,并且避免了多线程之间的上下文切换导致的开销。