《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网络编程中的 ...
 
随机推荐
- 【python-selenium】python-selenium安装配置
			
selenium 是一个web的自动化测试工具,不少学习功能自动化的同学开始首选selenium ,相因为它相比QTP有诸多有点: * 免费,也不用再为破解QTP而大伤脑筋 * 小巧,对于不同的语 ...
 - javascript 模块化编程
			
The module pattern is a common JavaScript coding pattern. It’s generally well understood, but there ...
 - PAT甲级 1129. Recommendation System (25)
			
1129. Recommendation System (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue ...
 - [ X.XX]CF每日一题系列线下更新中
			
现在做的都好水啊,就不写博客了,线下有记录滴
 - codeforces 455E
			
题目:http://codeforces.com/problemset/problem/455/E 题意:给定数组a,及f的定义: f[1][j] = a[j]; 1 <= j <= n ...
 - [zjoi2010]cheese
			
题目: 贪吃的老鼠(cheese.c/cpp/pas/in/out) 时限:每个测试点10秒 [问题描述] 奶酪店里最近出现了m只老鼠!它们的目标就是把生产出来的所有奶酪都吃掉.奶酪店中一天会生产n块 ...
 - AngularJS AOP 实例
			
AngularJS有种机制叫做拦截器(interceptor),它是$http扩展点,类似ASP.NET MVC的过滤器filter机制,对每个$http请求的发送和接收过程进行过滤. $httpPr ...
 - iOS笔记之UIKit_UILable
			
UILabel*label3 = [[UILabel alloc]initWithFrame:CGRectMake(0, 60+10+60+10+60+10, 320, 60)]; label3.ba ...
 - Class AB与Class D功放
			
D类功放 又称之为数字功放,其特点是,工作效率高,体积小. D类功放的结构 第一部分为调制器,最简单的只需用一只运放构成比较器即可完成.把原始音频信号加上一定直流偏置后放在运放的正输入 ...
 - 11i - 12 Gather Schema Statistics fails with Ora-20001 errors after 11G database Upgrade (文档 ID 781813.1)
			
11i - 12 Gather Schema Statistics fails with Ora-20001 errors after 11G database Upgrade (文档 ID 7818 ...