java 非阻塞_Java之NIO(非阻塞IO)

【1】NIO的与IO的区别:

总的来说java 中的IO 和NIO的区别主要有3点:

1)IO是面向流的,NIO是面向缓冲的;

2)IO是阻塞的,NIO是非阻塞的;

3)IO是单线程的,NIO 是通过选择器来模拟多线程的;

1. 通道

通道 Channel 是对原 I/O 包中的流的模拟,可以通过它读取和写入数据。

通道与流的不同之处在于,流只能在一个方向上移动(一个流必须是 InputStream 或者 OutputStream 的子类),而通道是双向的,可以用于读、写或者同时用于读写。

通道包括以下类型:

FileChannel:从文件中读写数据;

DatagramChannel:通过 UDP 读写网络中数据;

SocketChannel:通过 TCP 读写网络中数据;

ServerSocketChannel:可以监听新进来的 TCP 连接,对每一个新进来的连接都会创建一个 SocketChannel。

2. 缓冲区

发送给一个通道的所有数据都必须首先放到缓冲区中,同样地,从通道中读取的任何数据都要先读到缓冲区中。也就是说,不会直接对通道进行读写数据,而是要先经过缓冲区。

缓冲区实质上是一个数组,但它不仅仅是一个数组。缓冲区提供了对数据的结构化访问,而且还可以跟踪系统的读/写进程。

缓冲区包括以下类型:

ByteBuffer

CharBuffer

ShortBuffer

IntBuffer

LongBuffer

FloatBuffer

DoubleBuffer

缓冲区状态变量

capacity:最大容量;

position:当前已经读写的字节数;

limit:还可以读写的字节数。

状态变量的改变过程举例:

① 新建一个大小为 8 个字节的缓冲区,此时 position 为 0,而 limit = capacity = 8。capacity 变量不会改变,下面的讨论会忽略它。

dd05b079e308

image.png

② 从输入通道中读取 5 个字节数据写入缓冲区中,此时 position 移动设置为 5,limit 保持不变。

dd05b079e308

image.png

③ 在将缓冲区的数据写到输出通道之前,需要先调用 flip() 方法,这个方法将 limit 设置为当前 position,并将 position 设置为 0。

dd05b079e308

image.png

④ 从缓冲区中取 4 个字节到输出缓冲中,此时 position 设为 4。

dd05b079e308

image.png

⑤ 最后需要调用 clear() 方法来清空缓冲区,此时 position 和 limit 都被设置为最初位置。

dd05b079e308

image.png

3. 选择器

NIO 常常被叫做非阻塞 IO,主要是因为 NIO 在网络通信中的非阻塞特性被广泛使用。

NIO 实现了 IO 多路复用中的 Reactor 模型,一个线程 Thread 使用一个选择器 Selector 通过轮询的方式去监听多个通道 Channel 上的事件,从而让一个线程就可以处理多个事件。

通过配置监听的通道 Channel 为非阻塞,那么当 Channel 上的 IO 事件还未到达时,就不会进入阻塞状态一直等待,而是继续轮询其它 Channel,找到 IO 事件已经到达的 Channel 执行。

因为创建和切换线程的开销很大,因此使用一个线程来处理多个事件而不是一个线程处理一个事件,对于 IO 密集型的应用具有很好地性能。

应该注意的是,只有套接字 Channel 才能配置为非阻塞,而 FileChannel 不能,为 FileChannel 配置非阻塞也没有意义。

dd05b079e308

image.png

NIO在基础的IO流上发展处新的特点,分别是:内存映射技术,字符及编码,非阻塞I/O和文件锁定。下面我们分别就这些技术做一些说明

【2】NIO新特性:

内存映射技术,字符及编码,非阻塞I/O和文件锁定

<1>内存映射

这个功能主要是为了提高大文件的读写速度而设计的。内存映射文件(memory-mappedfile)能让你创建和修改那些大到无法读入内存的文件。有了内存映射文件,你就可以认为文件已经全部读进了内存,然后把它当成一个非常大的数组来访问了。将文件的一段区域映射到内存中,比传统的文件处理速度要快很多。内存映射文件它虽然最终也是要从磁盘读取数据,但是它并不需要将数据读取到OS内核缓冲区,而是直接将进程的用户私有地址空间中的一部分区域与文件对象建立起映射关系,就好像直接从内存中读、写文件一样,速度当然快了。

NIO中内存映射主要用到以下两个类:

[1] java.nio.MappedByteBuffer

[2] java.nio.channels.FileChannel

下面我们通过一个例子来看一下内存映射读取文件和普通的IO流读取一个大文件(文件大小为102603KB)的速度对比:

import java.io.RandomAccessFile;

import java.nio.ByteBuffer;

import java.nio.MappedByteBuffer;

import java.nio.channels.FileChannel;

/**

* @author lm

* @create 2018-09-09 8:32

* @desc NIO特性之内存映射

**/

public class MemoryMapper {

public static void main(String[] args) {

try {

RandomAccessFile file = new RandomAccessFile("F:\\IDEAWorkSpace\\AlgorithmTrain\\src\\com\\lm\\source\\triplets.txt","rw");

FileChannel channel = file.getChannel();

MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_ONLY,0,channel.size());

ByteBuffer buffer1 = ByteBuffer.allocate(1024);

byte[] b = new byte[1024];

long len = file.length();

long startTime = System.currentTimeMillis();

//读取内存映射文件

for(int i=0;i

if (len - i > 1024) {

buffer.get(b);

} else {

buffer.get(new byte[(int)(len - i)]);

}

}

long endTime = System.currentTimeMillis();

System.out.println("使用内存映射方式读取文件总耗时: "+(endTime - startTime));

//普通IO流方式

long startTime1 = System.currentTimeMillis();

while(channel.read(buffer1) > 0){ </

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值