深入理解Java AIO(三)—— Linux中的AIO实现
我们调用的Java AIO底层也是要调用OS的AIO实现,而OS主要也就Windows和Linux这两大类,当然还有Solaris和mac这些小众的。
在 Windows 操作系统中,提供了一个叫做 I/O Completion Ports 的方案,通常简称为 IOCP,操作系统负责管理线程池,其性能非常优异,所以在 Windows 中 JDK 直接采用了 IOCP 的支持。
而在 Linux 中其实也是有AIO 的实现的,但是限制比较多,性能也一般,所以 JDK 采用了自建线程池的方式,也就是说JDK并没有用Linux提供的AIO。但是本文主要想聊的,就是Linux中的AIO实现。
其实Linux的AIO简单来说是依赖epoll实现的伪AIO,一张图加一小段说明就能讲完。但是我们还是先讲一些前置知识来铺垫一下,最后再聊Linux的AIO。(当然本文不会深入到源码级别,只是讲大概的流程,要说为什么,当然是因为看C语言写的东西实在是太累了,特别是看源码,何况还可能混了汇编)
前置知识
缓存 I/O
缓存 I/O 又被称作标准 I/O,大多数文件系统的默认 I/O 操作都是缓存 I/O。在 Linux 的缓存 I/O 机制中,操作系统会将 I/O 的数据缓存在文件系统的页缓存( page cache )中,也就是说,数据会先被拷贝到操作系统内核的缓冲区中,然后才会从操作系统内核的缓冲区拷贝到应用程序的地址空间。
缓存的两个阶段
等待数据准备(等待数据进入内核缓存)
将数据从内核缓存拷贝到进程内存中
进程的阻塞