I/O的同步异步,阻塞非阻塞:
阻塞:当执行的操作所需的数据还没准备好时,线程进行等待
非阻塞:当数据还没准备好时,线程不等待
同步:执行操作,一直等操作执行完才向下执行
异步:执行操作,调用接口后不用等待,向下执行
常用的 read() 和 write() 方法都是同步I/O。
传统的I/O是阻塞式的。
问题:当操作所需的数据没有准备好,如数据没有到达,线程会一直等待。
解决:NIO(同步非阻塞I/O)
NIO相当于定义一个调度器selector,把所有实现通信的信道channel注册到selector,由selector调度各个channel的运行情况,实现各个channel不发生阻塞地传输。
Channel信道
FileChannel、DatagramChannel(UDP)、SocketChannel(TCP)、ServerSocketChannel(TCP)
这些通道涵盖了UDP 和 TCP 网络IO,以及文件IO。
Buffer缓冲
(有三个参数,容量capacity,当前存储位置position,上界limit)
写模式下,limit等于capacity;读模式下,limit等于position;
ByteBuffer、CharBuffer、DoubleBuffer、FloatBuffer、IntBuffer、LongBuffer、ShortBuffer
这些Buffer覆盖了你能通过IO发送的基本数据类型:byte, short, int, long, float, double 和 char。
Java NIO 还有个 MappedByteBuffer,用于表示内存映射文件。
Selector调度器
(可以比为一个调度工具)
Selector允许单线程处理多个 Channel,
用于向 buffer 提供数据或者读取 buffer 数据 ,buffer 对象的唯一接口。
过程:
1. 创建selector,channel,把channel信道设为非阻塞模式
2. 绑定socket对象到channel
3. 把channel注册到selector上
4. 调用selector的selectedKeys检查所有信道是否有需要的事情发生,如果有事情发生,返回所有的channel对象。
5. 通过信道,读取通信的数据,读取的是buffer。