NIO笔记(1)

NIO(同步非阻塞)基本架构

在这里插入图片描述

Selector、Channel、Buffer

  • 每一个Channel对应一个Buffer,Channel是双向的
  • 程序切换到Channel是由事件决定的,Event是一个重要的概念
  • Selector对应一个线程,一个线程对应多个Channel
  • Selector根据不同的事件,在各个通道上进行切换
  • Buffer就是一个内存块,是双向的,底层是有一个数组
  • 数据的读写是通过Buffer

理解非阻塞的概念(关注的是程序在等待调用结果(消息,返回值)时的状态)

从上图理解非阻塞:

  • 一个线程通过Selector去轮询Channel里面的任务,如果某一个Channel里面有读写事件,就去处理这个任务,但是如果所有的Channel都没有需要处理的任务,则这个线程可以去做其他的事情。

  • 一个线程请求写入一些数据到某通道,但是不需要等待它完全写入,这个线程可以去做其他的事情

比较异步和同步

  • 同步和异步关注的是消息通信机制
  • 所谓同步就是调用者进行调用后,在没有得到结果之前,该调用一直不会返回,但是一旦调用返回,就得到了返回值,同步就是指调用者主动等待调用结果;
  • 而异步则相反,执行调用之后直接返回,所以可能没有返回值,等到有返回值时,由被调用者通过状态,通知来通知调用者.异步就是指被调用者来通知调用者调用结果就绪
  • 所以,二者在消息通信机制上有所不同,一个是调用者检查调用结果是否就绪,一个是被调用者通知调用者结果就绪

NIO和BIO的一些区别

  • BIO是以流的方式来处理数据的,而NIO是以块(Buffer)的方式来处理数据块的,因为NIO的效率比BIO高
  • BIO是阻塞的,NIO是非阻塞的
  • BIO是基于字节流和字符流进行操作的,而NIO是基于Channel(通道)和Buffer(缓冲区)进行操作的,数据总是从通道读取数据到缓冲区,或者从缓冲区写入到通道中。每个线程可以监听多个客户端的连接,而BIO需要建立和客户端数量一样多的线程。

Unix的I/O类型

  • 阻塞I/O(bloking IO)

  • 非阻塞I/O(nonblocking IO)

  • 多路复用I/O(IO multiplexing)

  • 信号驱动I/O(signal driven IO)在这里插入图片描述

  • 异步I/O(asynchronous IO)

前4种都是同步,只有最后一种是异步I/O.需要注意的是Java NIO依赖于Unix系统的多路复用I/O,对于I/O操作来说,它是同步I/O

  • 阻塞I/O(bloking IO)
     阻塞I/O

  • 非阻塞I/O(nonblocking IO)
    在这里插入图片描述

  • 多路复用I/O(IO multiplexing)
    在这里插入图片描述

  • 信号驱动I/O(signal driven IO)

在这里插入图片描述

  • 信号驱动I/O(signal driven IO)
    在这里插入图片描述

多路复用I/O

与多进程和多线程技术相比,I/O多路复用技术的最大优势是系统开销小,系统不必创建进程/线程,也不必维护这些进程/线程,从而大大减小了系统的开销。

I/O多路复用是指使用一个线程调用select和poll函数来检查多个文件描述符(Socket)的就绪状态,比如调用select和poll函数,传入多个文件描述符,如果有一个文件描述符就绪,则返回,否则阻塞直到超时

目前支持I/O多路复用的系统调用有 select,pselect,poll,epoll

select、poll、epoll简介

  • select

函数监视的文件描述符分3类,分别是writefds、readfds、和exceptfds。调用后select函数会阻塞,直到有描述符就绪(有数据 可读、可写、或者有except),或者超时(timeout指定等待时间,如果立即返回设为null即可),函数返回。当select函数返回后,可以通过遍历fdset,来找到就绪的描述符。

  • poll

poll本质上和select没有区别,它将用户传入的数组拷贝到内核空间,然后查询每个fd对应的设备状态,如果设备就绪则在设备等待队列中加入一项并继续遍历,如果遍历完所有fd后没有发现就绪设备,则挂起当前进程,直到设备就绪或者主动超时,被唤醒后它又要再次遍历fd。这个过程经历了多次无谓的遍历。

  • epoll

epoll是在2.6内核中提出的,是之前的select和poll的增强版本。相对于select和poll来说,epoll更加灵活,没有描述符限制。epoll使用一个文件描述符管理多个描述符,将用户关系的文件描述符的事件存放到内核的一个事件表中,这样在用户空间和内核空间的copy只需一次。

小结

JAVA中的NIO对应linux中的IO多路复用,Reactor是一个使用了同步非阻塞的I/O多路复用机制的模式

参考文档:

### 关于尚硅谷 Java NIO 学习笔记或 PDF 的查找 目前,尚未提供具体的尚硅谷 Java NIO 学习笔记或 PDF 文件的相关链接。然而,可以通过以下方式获取相关资源: 1. **官方渠道** 尚硅谷作为一家知名的 IT 教育机构,通常会在其官方网站或其他授权平台上发布课程资料。建议访问尚硅谷官网,搜索与 Java NIO 相关的课程,并下载配套的学习资料[^1]。 2. **第三方平台** 许多教育分享网站可能提供了尚硅谷的 Java NIO 学习笔记或视频教程。可以在诸如 优快云、GitHub 或者百度网盘等平台上尝试搜索关键词“尚硅谷 Java NIO”,找到对应的 PDF 或其他格式的文档[^2]。 3. **核心知识点总结** 如果暂时无法获得完整的 PDF 资料,以下是基于 Java NIO 的基础知识整理,供参考: #### 什么是 NIOJava NIO 是 New IO 的缩写,它是从 JDK 1.4 版本开始引入的一套新的 I/O API。相比传统 I/O,Java NIO 提供了更高效的非阻塞模式以及缓冲区机制。它主要包括以下几个部分: - Buffer(缓冲区) - Channel(通道) - Selector(选择器) #### 核心组件详解 - **Buffer**: 数据容器,用于存储不同类型的字节数据。常用的方法包括 `flip()`、`clear()` 和 `rewind()`. 使用时需注意缓冲区的状态管理[^5]。 ```java ByteBuffer buffer = ByteBuffer.allocate(1024); buffer.put((byte) 'A'); buffer.flip(); while (buffer.hasRemaining()) { System.out.print((char) buffer.get()); } ``` - **Channel**: 类似于流的概念,但支持双向读写操作。常见实现包括 `FileChannel` 和 `SocketChannel`. ```java try (RandomAccessFile file = new RandomAccessFile("example.txt", "rw"); FileChannel channel = file.getChannel()) { ByteBuffer buf = ByteBuffer.allocate(1024); int bytesRead = channel.read(buf); // 从通道读取数据到缓冲区 buf.flip(); // 切换为读模式 while (buf.hasRemaining()) { System.out.print((char) buf.get()); } } catch (IOException e) { e.printStackTrace(); } ``` - **Selector**: 多路复用的关键工具,允许单线程处理多个 Channel。适用于高并发场景中的网络通信[^4]. #### 如何验证学习效果? 通过编写简单的代码测试对 NIO 的理解程度。例如,利用 `FileChannel` 实现文件拷贝功能: ```java import java.io.*; import java.nio.channels.FileChannel; public class FileCopyExample { public static void main(String[] args) throws IOException { FileInputStream fis = new FileInputStream("source.txt"); FileOutputStream fos = new FileOutputStream("destination.txt"); FileChannel sourceChannel = fis.getChannel(); FileChannel destChannel = fos.getChannel(); long bytesTransferred = sourceChannel.transferTo(0, sourceChannel.size(), destChannel); sourceChannel.close(); destChannel.close(); fis.close(); fos.close(); System.out.println(bytesTransferred + " bytes copied."); } } ``` ---
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值