《Netty权威指南》(二)NIO 入门
[TOC]
2.1 同步阻塞 I/O
采用 BIO 通信模型的服务器,通常由一个独立的 Acceptor 线程负责监听客户端的连接,它接收到客户端连接请求之后为每个客户端创建一个新的线程进行处理,处理完成后,通过输出流返回应答给客户端,线程销毁。
A1[Socket] -->|读/写| B(Acceptor 线程)
A2[Socket] -->|读/写| B
A3[Socket] -->|读/写| B
B -->|创建| C1[Thread]
B -->|创建| C2[Thread]
B -->|创建| C3[Thread]
2.2 伪异步 I/O
采用 线程池 和 任务队列 可以实现一种叫做 伪异步 I/O 通信框架。 当有新的客户端接入时,将客户端的 Socket 封装成一个 Task 投递到后端的线程池中进行处理,JDK 的线程池维护着一个消息队列和 N 个活跃线程,对消息队列中的任务进行处理。由于线程池可以设置消息队列的大小和最大线程数,因此它的资源占用时可控的,无论多少个客户端并发访问,都不会导致资源的耗尽和宕机。
A1[Socket] -->|读/写| B(Acceptor 线程)
A2[Socket] -->|读/写| B
A3[Socket] -->|读/写| B
B --> |提交 Task|P[Thread Pool]
B --> |提交 Task|P
B --> |提交 Task|P
2.3 NIO
NIO 官方称为 New I/O,目标是要让 Java 支持非阻塞 I/O,所以通常也叫非阻塞 I/O(Non-blocking I/O)。
阻塞模式使用非常简单,但是性能和可靠性都不好,非阻塞模式则正好相反。一般来说,低负载、低并发的应用程序可以选择同步阻塞 I/O 以降低编程复杂度;对于高负载、高并发的网络应用,需要使用 NIO 的非阻塞模式进行开发。
A1[Socket] -->|读/写| B(Selector)
A2[Socket] -->|读/写| B
A3[Socket] -->|读/写| B
B --> P[Thread]
2.3.1 Buffer、Channel、Selector
1. 缓冲区 Buffer
Buffer 是一个对象,它包含一些要写入或者要读出的数据。在 NIO 库中,所有数据都是用缓冲区处理的。在读取数据时,它是直接从缓冲区读取;在写入数据时,写入到缓冲区。
2. 通道 Channel
网络数据通过 Channel 读取和写入。通道与流的不同之处在于通道是双向的,流只是在一个方向移动,而通道可用于读、写或者二者同时进行。
因为 Channel 是全双工的,所以它可以比流更好地映射底层操作系统的 API。
3. 多路复用器 Selector
多路复用器提供选择已经就绪的任务的能力。Selector 会不断地轮询注册在其上的 Channel,如果某个 Channel 上面发生读或者写事件,这个 Channel 就处于就绪状态,会被 Selector 轮询出来,然后通过 SelectionKey 可以获取就绪 Channel 的集合,进行后续的 I/O 操作。
2.3.2 NIO 服务端序列图

(1)打开 ServerSocketChannel
(2)绑定监听地址 InetSocketAddress
(3)创建 Selector,启动线程
(4)将 ServerSocketChannel 注册到 Selector,监听 ACCEPT
(5)Selector 轮询注册的 Key
(6)handleAccept() 处理新的客户端接入
(7)设置新建客户端连接的 Socket 参数
(8)向 Selector 注册监听读操作 SelectionKey.OP_READ
(9)handleRead() 异步请求消息到 ByteBuffer,收到请求
(10)decode 请求消息
(11)异步写 ByteBuffer 到 SocketChannel,发送响应
2.3.1 NIO 客户端序列图

(1)打开 SocketChannel
(2)设置 SocketChannel 为非阻塞模式,同时设置 TCP 参数
(3)异步连接服务端
(4)判断连接结果,如果连接成功,调到步骤10,否则执行步骤5
(5)向 Reactor 线程的多路复用器注册 OP_CONNECT 事件
(6)创建 Selector,启动线程
(7)Selector 轮询就绪的 Key
(8)handleConnect()
(9)判断连接是否完成,完成执行步骤10
(10)向多路复用器注册读事件 OP_READ
(11)encode 请求消息
(12)异步写 ByteBuffer 到 SocketChannel,发送请求
(13)handleRead() 异步从 SocketChannel 读 ByteBuffer,收到响应
2.4 AIO
NIO 2.0 引入了新的异步通道的概念,并提供了异步文件通道和异步套接字通道的实现。异步通道提供以下两种方式获取操作结果。
- 通过 java.util.concurrent.Future 类表示异步操作的结果;
- 在执行异步操作的时候传入一个 java.nio.channels。
CompletionHandler 接口的实现类作为操作完成的回调。
NIO 2.0 的异步套接字通道是真正的异步非阻塞 I/O,对应于 UNIX 网络编程中的事件驱动 I/O(AIO)。它不需要通过多路复用器(Selector)对注册的通道进行轮询操作即可实现异步读写,从而简化了 NIO 的编程模型。
2.5 4 种 I/O 的对比
| 同步阻塞I/O | 伪异步I/O | 非阻塞I/O(NIO) | 异步I/O(AIO) | |
|---|---|---|---|---|
| 客户端个数:I/O线程 | 1:1 | M:N | M:1 | M:0 |
| I/O(阻塞)类型 | 阻塞 | 阻塞 | 非阻塞 | 非阻塞 |
| I/O(同步)类型 | 同步 | 同步 | 同步(I/O多路复用) | 异步 |
| API使用难度 | 简单 | 简单 | 复杂 | 一般 |
| 调试难度 | 简单 | 简单 | 复杂 | 复杂 |
| 可靠性 | 非常差 | 非常差 | 高 | 高 |
| 吞吐量 | 低 | 中 | 高 | 高 |
2.6 选择 Netty 的理由
什么是 Netty
Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients.
Netty 是一款异步的事件驱动的网络应用程序框架,支持快速地开发可维护的高性能的面向协议的服务器和客户端。
Netty 的优点
- 功能强大
- 使用简单
- 性能高
- 安全
- 社区活跃
《Netty权威指南》(二)NIO 入门的更多相关文章
- Netty权威指南之NIO通信模型
NIO简介:与Socket和ServerSocket类相对应,NIO提供了SocketChannel和ServerSocketChannel两种不同的套接字通道实现,这两种新通道都支持阻塞和非阻塞两种 ...
- Netty权威指南
Netty权威指南(异步非阻塞通信领域的经典之作,国内首本深入剖析Netty的著作,全面系统讲解原理.实战和源码,带你完美进阶Netty工程师.) 李林锋 著 ISBN 978-7-121-233 ...
- 《Netty权威指南》
<Netty权威指南> 基本信息 作者: 李林锋 出版社:电子工业出版社 ISBN:9787121233432 上架时间:2014-5-29 出版日期:2014 年6月 开本:16开 页码 ...
- 《Netty 权威指南(第2 版)》目录
图书简介:<Netty 权威指南(第2 版)>是异步非阻塞通信领域的经典之作,基于最新版本的Netty 5.0 编写,是国内很难得一见的深入介绍Netty 原理和架构的书籍,也是作者多年实 ...
- netty权威指南学习笔记二——netty入门应用
经过了前面的NIO基础知识准备,我们已经对NIO有了较大了解,现在就进入netty的实际应用中来看看吧.重点体会整个过程. 按照权威指南写程序的过程中,发现一些问题:当我们在定义handler继承Ch ...
- 《Netty权威指南》目录
一.基础篇 走进Java NIO 1. Java 的 I/O 演进之路:https://www.cnblogs.com/zengzhihua/p/9930652.html 2. NIO 入门:http ...
- netty权威指南学习笔记六——编解码技术之MessagePack
编解码技术主要应用在网络传输中,将对象比如BOJO进行编解码以利于网络中进行传输.平常我们也会将编解码说成是序列化/反序列化 定义:当进行远程跨进程服务调用时,需要把被传输的java对象编码为字节数组 ...
- 《Netty权威指南》笔记
第1章 Java的I/O演进之路 1.1 Linux网络I/O模型 fd:file descriptor,文件描述符.linux内核将所有外部设备都看作一个文件来操作,对文件的读写会调用内核提供的命令 ...
- Netty权威指南(笔记一)
转载:http://blog.csdn.net/clarkkentyang/article/details/52529785 第一章(略) 第二章 NIO入门 2.1传统的BIO编程(同步阻塞I/O服 ...
- Netty权威指南之AIO编程
由JDK1.7提供的NIO2.0新增了异步的套接字通道,它是真正的异步I/O,在异步I/O操作的时候可以传递信号变量,当操作完成后会回调相关的方法,异步I/o也被称为AIO,对应于UNIX网络编程中的 ...
随机推荐
- (求凹包) Bicycle Race (CF 659D) 简单题
http://codeforces.com/contest/659/problem/D Maria participates in a bicycle race. The speedway t ...
- 2016-2017-2 20155326实验二《Java面向对象程序设计》实验报告
2016-2017-2 20155326实验二<Java面向对象程序设计>实验报告 实验内容 初步掌握单元测试和TDD 理解并掌握面向对象三要素:封装.继承.多态 初步掌握UML建模 熟悉 ...
- STL-容器库000
容器库已经作为class templates 实现. 容器库中是编程中常用的结构: (1)动态数组结构vector: (2)队列queue: (3)栈stack: (4)heaps 堆priority ...
- 数组中超过N分之一的数字
寻找数组中超过一半的元素,这是一道十分经典和普遍的面试题了,实现起来比较容易,只是需要写技巧,将问题扩展就可以衍生到求数组中几个超过N分一的元素,例如找出数组中3个出现次数超过1/4的元素. /*** ...
- spring boot mybatis sql打印到控制台
如何设置spring boot集成 mybatis 然后sql语句打印到控制台,方便调试: 设置方法: 在application.properties文件中添加: logging.level.com. ...
- Python自动化开发 - 字符编码、文件和集合
本节内容 字符编码 文件操作 集合 一.字符编码 1.编码 计算机只能处理数字,如果要处理文本,就必须先把文本转换为数字才能处理.解决思路:数字与符号建立一对一映射,用不同数字表示不同符号. ASCI ...
- VARCHAR的最大长度的问题
大家知道,在SQL Server 2000中,VARCHAR的最大长度是8000,如果字符串的长度超过8000,保存在VARCHAR中时就会被截断.如果你需要传入的参数恰好很长,比如是一个xml,很多 ...
- C++动态(显式)调用 C++ dll
1.创建DLL新项目Dll1,Dll1.cpp: extern "C" __declspec(dllexport) const char* myfunc() { return &q ...
- WPF 定义Lookless控件的默认样式、 OnApplyTemplate 如何使用(实现方式、如何工作的)!
写的非常详细: 作者地址:https://www.cnblogs.com/atskyline/archive/2012/11/16/2773806.html 参考资料: http://www.code ...
- Could not find installable ISAM
程序中去读EXCEL文档,以前一直参考<Asp.net读取Excel文件 2>http://www.cnblogs.com/insus/archive/2011/05/05/2037808 ...