你好,我是彤哥,本篇是netty系列的第三篇。

欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识。

简介

上一章我们介绍了IO的五种模型,实际上Java只支持其中的三种,即BIO/NIO/AIO。

本文将介绍Java中这三种IO的进化史,并从使用的角度剖析它们背后的故事。

Java BIO

BIO概念解析

BIO,Blocking IO,阻塞IO,它是Java的上古产品,自出生就有的东西(JDK 1.0)。

使用BIO则数据准备和数据从内核空间拷贝到用户空间两个阶段都是阻塞的。

BIO使用案例

public class EchoServer {

public static void main(String[] args) throws IOException {

ServerSocket serverSocket = new ServerSocket(8080);

while (true) {

System.out.println("start accept");

Socket socket = serverSocket.accept();

System.out.println("new conn: " + socket.getRemoteSocketAddress());

new Thread(()->{

try {

BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream()));

String msg;

// 读取消息,本文来源公从号彤哥读源码

while ((msg = reader.readLine()) != null) {

if (msg.equalsIgnoreCase("quit")) {

reader.close();

socket.close();

break;

} else {

System.out.println("receive msg: " + msg);

}

}

} catch (IOException e) {

e.printStackTrace();

}

}).start();

}

}

}

客户端可以使用telnet来测试,而且你可以使用多个telnet来测试:

[c:\~]$ telnet 127.0.0.1 8080

Connecting to 127.0.0.1:8080...

Connection established.

To escape to local shell, press 'Ctrl+Alt+]'.

hello world

我是人才

quit

Connection closed by foreign host.

BIO的使用方式非常简单,服务端接收到一个连接就启动一个线程来处理这个连接的所有请求。

所以,BIO最大的缺点就是浪费资源,只能处理少量的连接,线程数随着连接数线性增加,连接越多线程越多,直到抗不住。

Java NIO

NIO概念解析

NIO,New IO,JDK1.4开始支持,内部是基于多路复用的IO模型。

这里有个歧义,很多人认为Java的NIO是Non-Blocking IO的缩写,其实并不是。

使用NIO则多条连接的数据准备阶段会阻塞在select上,数据从内核空间拷贝到用户空间依然是阻塞的。

因为第一阶段并不是连接本身处于阻塞阶段,所以通常来说NIO也可以看作是同步非阻塞IO。

NIO使用案例

public class EchoServer {

public static void main(String[] args) throws IOException {

Selector selector = Selector.open();

ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();

serverSocketChannel.bind(new InetSocketAddress(8080));

serverSocketChannel.configureBlocking(false);

// 将accept事件绑定到selector上

serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);

while (true) {

// 阻塞在select上

selector.select();

Set selectionKeys = selector.selectedKeys();

// 遍历selectKeys

Iterator iterator = selectionKeys.iterator();

while (iterator.hasNext()) {

SelectionKey selectionKey = iterator.next();

// 如果是accept事件

if (selectionKey.isAcceptable()) {

ServerSocketChannel ssc = (ServerSocketChannel) selectionKey.channel();

SocketChannel socketChannel = ssc.accept();

System.out.println("accept new conn: " + socketChannel.getRemoteAddress());

socketChannel.configureBlocking(false);

socketChannel.register(selector, SelectionKey.OP_READ);

} else if (selectionKey.isReadable()) {

// 如果是读取事件,本文来源公从号彤哥读源码

SocketChannel socketChannel = (SocketChannel) selectionKey.channel();

ByteBuffer buffer = ByteBuffer.allocate(1024);

// 将数据读入到buffer中

int length = socketChannel.read(buffer);

if (length > 0) {

buffer.flip();

byte[] bytes = new byte[buffer.remaining()];

// 将数据读入到byte数组中

buffer.get(bytes);

// 换行符会跟着消息一起传过来

String content = new String(bytes, "UTF-8").replace("\r\n", "");

if (content.equalsIgnoreCase("quit")) {

selectionKey.cancel();

socketChannel.close();

} else {

System.out.println("receive msg: " + content);

}

}

}

iterator.remove();

}

}

}

}

这里同样使用telnet测试,而且你可以使用多个telnet来测试:

[c:\~]$ telnet 127.0.0.1 8080

Connecting to 127.0.0.1:8080...

Connection established.

To escape to local shell, press 'Ctrl+Alt+]'.

hello world

我是人才

quit

Connection closed by foreign host.

NIO的使用方式就有点复杂了,但是一个线程就可以处理很多连接。

首先,需要注册一个ServerSocketChannel并把它注册到selector上并监听accept事件,然后accept到连接后会获取到SocketChannel,同样把SocketChannel也注册到selector上,但是监听的是read事件。

NIO最大的优点,就是一个线程就可以处理大量的连接,缺点是不适合处理阻塞性任务,因为阻塞性任务会把这个线程占有着,其它连接的请求将得不到及时处理。

Java AIO

AIO概念介绍

AIO,Asynchronous IO,异步IO,JDK1.7开始支持,算是一种比较完美的IO,Windows下比较成熟,但Linux下还不太成熟。

使用异步IO则会在请求时立即返回,并在数据已准备且已拷贝到用户空间后进行回调处理,两个阶段都不会阻塞。

AIO使用案例

public class EchoServer {

public static void main(String[] args) throws IOException {

AsynchronousServerSocketChannel serverSocketChannel = AsynchronousServerSocketChannel.open();

serverSocketChannel.bind(new InetSocketAddress(8080));

// 监听accept事件,本文来源公从号彤哥读源码

serverSocketChannel.accept(null, new CompletionHandler() {

@Override

public void completed(AsynchronousSocketChannel socketChannel, Object attachment) {

try {

System.out.println("accept new conn: " + socketChannel.getRemoteAddress());

// 再次监听accept事件

serverSocketChannel.accept(null, this);

// 消息的处理

while (true) {

ByteBuffer buffer = ByteBuffer.allocate(1024);

// 将数据读入到buffer中

Future future = socketChannel.read(buffer);

if (future.get() > 0) {

buffer.flip();

byte[] bytes = new byte[buffer.remaining()];

// 将数据读入到byte数组中

buffer.get(bytes);

String content = new String(bytes, "UTF-8");

// 换行符会当成另一条消息传过来

if (content.equals("\r\n")) {

continue;

}

if (content.equalsIgnoreCase("quit")) {

socketChannel.close();

break;

} else {

System.out.println("receive msg: " + content);

}

}

}

} catch (Exception e) {

e.printStackTrace();

}

}

@Override

public void failed(Throwable exc, Object attachment) {

System.out.println("failed");

}

});

// 阻塞住主线程

System.in.read();

}

}

这里同样使用telnet测试,而且你可以使用多个telnet来测试:

[c:\~]$ telnet 127.0.0.1 8080

Connecting to 127.0.0.1:8080...

Connection established.

To escape to local shell, press 'Ctrl+Alt+]'.

hello world

我是人才

quit

Connection closed by foreign host.

AIO的使用方式不算太复杂,默认会启一组线程来处理用户的请求,而且如果在处理阻塞性任务,还会自动增加新的线程来处理其它连接的任务。

首先,创建一个AsynchronousServerSocketChannel并调用其accept方法,这一步相当于监听了accept事件,在收到accept事件后会获取到AsynchronousSocketChannel,然后就可以在回调方法completed()里面读取数据了,当然也要继续监听accept事件。

AIO最大的优点,就是少量的线程就可以处理大量的连接,而且可以处理阻塞性任务,但不能大量阻塞,否则线程数量会膨胀。

槽点

(1)三种IO的实现方式中对于换行符的处理竟然都不一样,BIO中不会把换行符带过来(其实是带过来了,因为用了readLine()方法,所以换行符没了),NIO中会把换行符加在消息末尾,AIO中会把换行符当成一条新的消息传过来,很神奇,为啥不统一处理呢,也很疑惑。

(2)JDK自带的ByteBuffer是一个难用的东西。

总结

本文我们从概念和使用两个角度分别介绍了BIO/NIO/AIO三种IO模型。

问题

看起来JDK的实现似乎很完美啊,为什么还会有Netty呢?

最后,也欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识。

java aio nio bio_3. 彤哥说netty系列之Java BIO NIO AIO进化史相关推荐

  1. java nio attachment_7. 彤哥说netty系列之Java NIO核心组件之Selector

    --日拱一卒,不期而至! 你好,我是彤哥,本篇是netty系列的第七篇. 简介 上一章我们一起学习了Java NIO的核心组件Buffer,它通常跟Channel一起使用,但是它们在网络IO中又该如何 ...

  2. java nio netty 教程,4. 彤哥说netty系列之Java NIO实现群聊(自己跟自己聊上瘾了),netty实现...

    4. 彤哥说netty系列之Java NIO实现群聊(自己跟自己聊上瘾了),netty实现 你好,我是彤哥,本篇是netty系列的第四篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识 ...

  3. java channel源码_彤哥说netty系列之Java NIO核心组件之Channel

    你好,我是彤哥,本篇是netty系列的第五篇. 欢迎来我的工从号彤哥读源码系统地学习源码&架构的知识. 简介 上一章我们一起学习了如何使用Java原生NIO实现群聊系统,这章我们一起来看看Ja ...

  4. java channel源码_5. 彤哥说netty系列之Java NIO核心组件之Channel

    你好,我是彤哥,本篇是netty系列的第五篇. 简介 上一章我们一起学习了如何使用Java原生NIO实现群聊系统,这章我们一起来看看Java NIO的核心组件之一--Channel. 思维转变 首先, ...

  5. 4. 彤哥说netty系列之Java NIO实现群聊(自己跟自己聊上瘾了)

    你好,我是彤哥,本篇是netty系列的第四篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 上一章我们一起学习了Java中的BIO/NIO/AIO的故事,本章将带着大家一起使 ...

  6. java 仿qq庅,4. 彤哥说netty系列之Java NIO实现群聊(自己跟自己聊上瘾了)

    你好,我是彤哥,本篇是netty系列的第四篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 上一章我们一起学习了Java中的BIO/NIO/AIO的故事,本章将带着大家一起使 ...

  7. 6. 彤哥说netty系列之Java NIO核心组件之Buffer

    --日拱一卒,不期而至! 你好,我是彤哥,本篇是netty系列的第六篇. 简介 上一章我们一起学习了Java NIO的核心组件Channel,它可以看作是实体与实体之间的连接,而且需要与Buffer交 ...

  8. 1. 彤哥说netty系列之开篇(有个问卷调查)

    你好,我是彤哥,本篇是netty系列的第一篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 本文主要讲述netty系列的整体规划,并调查一下大家喜欢的学习方式. 知识点 ne ...

  9. 2. 彤哥说netty系列之IO的五种模型

    你好,我是彤哥,本篇是netty系列的第二篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 本文将介绍linux中的五种IO模型,同时也会介绍阻塞/非阻塞与同步/异步的区别. ...

最新文章

  1. [JZOJ P1291] [DP]添加括号
  2. 简单团队-爬虫豆瓣top250-项目总结
  3. 对实体 quot;useSSLquot; 的引用必须以 ';' 分隔符结尾
  4. 我,宇宙最强编辑器,支持远程开发
  5. i3、i5、i7有什么区别?
  6. 8月20日全球六大国际域名解析量变化情况统计报告
  7. 新加坡金融管理局主席:数字货币会使传统风险变得更加明显
  8. GitHub 推出安全新功能,帮助开源软件发现漏洞和机密信息
  9. CodeSmith 创建Ado.Net自定义模版(一)
  10. JavaScript:获取帧率FrameUtil.js
  11. Requirement-Driven Linux Shell Programming
  12. 个人博客前后台整站开发——模板免费下载
  13. mysql怎么设主键和外键_MySQL基础之 主键外键设置
  14. 策略研集合竞选(附源码)
  15. 超市结算系统|Springboot+Vue通用超市结算收银系统
  16. c语言反三角函数值域,反三角函数的定义域和值域
  17. 如何解决eclipse黑底白字快速需求
  18. node中使用es6语法
  19. Python实战项目:高血压检测项目调查问卷接口的测试
  20. 笔记本计算机充不上电,笔记本充不了电怎么回事_笔记本电脑怎么充不了电-win7之家...

热门文章

  1. WampServer图标为黄色,无法启动的解决办法及思路
  2. 快讯:我国高中数学课本即将发生全面变革,微积分下放高中不是梦!
  3. Faiss 相似度搜索使用余弦相似性
  4. (转)为Linux服务器部署高效防毒软件
  5. 顾城其实很可怜 舒婷回忆:他一辈子都在为钱发愁
  6. 从0开始制作电子病历
  7. 三个步骤,让你迅速掌握专业人士的思维框架
  8. Windows一键启动jar包bat脚本制作
  9. postgresql 官网学习文档
  10. UI界面布局有哪些?