概述
JAVA NIO是IO复用模型( POSIX定义的五种IO模型: 阻塞IO, 非阻塞IO, IO复用, 异步IO, 信号驱动IO )的实现.
在Linux系统中,复用IO用三种实现: select, poll, epoll.
select
select实现有2个问题,文件描述符的限制以及需要对所有监听的channel进行遍历
poll
poll实现解决的是描述符限制, 依然还需要进行channel遍历
epoll
epoll解决了遍历channel的问题, 通过向内核注册事件, 当有监听事件发生时, 可以准确知道是哪个channel
复用IO本质上还是阻塞的, 内核读取到数据后, 用户进程从内核进程中拷贝数据的过程是同步阻塞的. IO模型中的异步I是完全异步的, 内核会将数据拷贝到用户进程后, 才会通知用户进程.
Selector
Java文档中描述Selector:
A multiplexor of SelectableChannel objects.
可以理解为SelectableChannel对象的多路复用实现.Channel(SelectableChannel)可以通过Selector实现多路复用.
java.nio中定义了一个Selector类,可以通过其open()静态方法获取Selector实例.
还有另外一种机制是通过SelectorProvider自定义实现获取selector, SelectorProvider是通过SPI机制动态发现其实现类.
Buffer
Buffer是一个数据缓冲区, 属于堆外内存, 不属于在JVM运行时内存模型.
Java文档中描述Buffer:
A container for data of a specific primitive type.
可以理解为基本类型数据的容器。
在输入的时候, Buffer是Channel的输入源, 在输出的时候, Buffer是输出源.
java.nio中实现多种基本类型的Buffer(ByteBuffer, CharBuffer, DoubleBuffer, FloatBuffer, IntBuffer, LongBuffer, ShortBuffer)
.
当类型是ByteBuffer的时候, 要注意编码的问题.
Channel
Java文档中对Channel的描述是:
A nexus for I/O operations.
可以理解为 IO操作的一个纽带.
java.nio.channels.SelectableChannel
定义了Channel可以通过selector实现多路复用.
DatagramChannel, Pipe.SinkChannel, Pipe.SourceChannel, ServerSocketChannel, SocketChannel
这几个类是其子类(实际是AbstractSelectableChannel的直接子类),也就是说这几个类型的channel可以实现多路复用.然而FileChannel不在其中, 也就是磁盘IO不能实现多路复用.
那么问题来了, FileChannel为什么不能实现多路复用?既然不能实现IO复用, 为何又要提供一个FileChannel的类呢?
参考资料
- NIO系列