5. 彤哥说netty系列之Java NIO核心组件之Channel

你好,我是彤哥,本篇是netty系列的第五篇。
简介
上一章我们一起学习了如何使用Java原生NIO实现群聊系统,这章我们一起来看看Java NIO的核心组件之一——Channel。
思维转变
首先,我想说的最重要的一个点是,学习NIO思维一定要从BIO那种一个连接一个线程的模式转变成多个连接(Channel)共用一个线程来处理的这种思维。


1个Connection = 1个Socket = 1个Channel,这几个概念可以看作是等价的,都表示一个连接,只不过是用在不同的场景中。
如果单从阻塞/非阻塞的角度来看的话,IO可以分成两大类,一类是Blocking IO,一类是Non-blocking IO,像IO五种模型中的后四种其实都可以看作是非阻塞型IO,只是各自使用的手段不相同罢了。
在Java中,我们说的非阻塞IO或者说NIO(New IO)主要是指多路复用IO,底层可以使用select/poll/epoll等技术实现。
另外,在Java1.7的时候引入了NIO2,这个主要是指异步IO模型,也就是我们常说的AIO,底层完全使用异步回调的方式来实现。
但是,由于AIO这项技术在linux操作系统上还不太成熟,所以我们通常也不会说太多关于这方面的内容。
在后面我们学习Netty的时候会再次讲到这三种IO模型,可以看到Netty是完全支持三种IO的,但是它把OIO(BIO)和AIO都给deprecated了,也进一步说明了AIO的不成熟性。
好了,扯了一下思维转变的问题,下面正式进入今天的内容——Java NIO核心组件之Channel。
Channel
概念
我们先来看看Java中对于Channel的定义,位于java.nio.channels.Channel类的注释上:
A nexus for I/O operations.
本文来源工从号彤哥读源码
A channel represents an open connection to an entity such as a hardware device, a file, a network socket, or a program component that is capable of performing one or more distinct I/O operations, for example reading or writing.
第一句,它是IO操作的一种连接。
nexus, the means of connection between things linked in series.
第二句,Channel代表到实体的开放连接,这个实体可以是硬件,文件,网络套接字,或者程序组件,并且可以执行一个或多个不同的IO操作,例如,读或写。
简单点讲,Channel就是实体与实体之间的连接,比如,操作文件可以使用FileChannel,操作网络可以使用SocketChannel等。
与流的区别
BIO是面向流(Stream)编程的,流又分成InputStream和OutputStream,那么Channel和Stream有什么区别呢?
Channel可以同时支持读和写,而Stream只能支持单向的读或写(所以分成InputStream和OutputStream)
Channel支持异步读写,Stream通常只支持同步
Channel总是读向(read into)Buffer,或者写自(write from)Buffer(有点绕,以Channel为中心,从Channel中读出数据到Buffer,从Buffer中往Channel写入数据)

实现方式
下面列举了JDK中比较重要的实现方式:
- FileChannel:操作文件
- DatagramChannel:UDP协议支持
- SocketChannel:TCP协议支持
- ServerSocketChannel:监听TCP协议Accept事件,之后创建SocketChannel
例子
public class FileChannelTest {
public static void main(String[] args) throws IOException {
// 从文件获取一个FileChannel
FileChannel fileChannel = new RandomAccessFile("D:\\object.txt", "rw").getChannel();
// 声明一个Byte类型的Buffer
ByteBuffer buffer = ByteBuffer.allocate(10);
// 将FileChannel中的数据读出到buffer中,-1表示读取完毕
// buffer默认为写模式,本文来源工从号彤哥读源码
// read()方法是相对channel而言的,相对buffer就是写
while ((fileChannel.read(buffer)) != -1) {
// buffer切换为读模式
buffer.flip();
// buffer中是否有未读数据
while (buffer.hasRemaining()) {
// 未读数据的长度
int remain = buffer.remaining();
// 声明一个字节数组
byte[] bytes = new byte[remain];
// 将buffer中数据读出到字节数组中
buffer.get(bytes);
// 打印出来
System.out.println(new String(bytes, StandardCharsets.UTF_8));
}
// 清空buffer,为下一次写入数据做准备
// clear()会将buffer再次切换为写模式
buffer.clear();
}
}
}
可以看到,Channel与Buffer是息息相关的。注意这里的读写方法,调用者是谁就以谁为核心,channel.read()就表示从channel读出数据,buffer.get()就表示从buffer读出数据,这跟传统编程的角度不太一样的地方。
总结
今天我们学习了Java NIO核心组件之Channel,它与传统BIO中的流很类似但又有所区别,且经常与Buffer联合起来使用,Buffer又是什么呢?请听下回分解。
参考
挺不错的一个网站:
http://tutorials.jenkov.com/java-nio/channels.html
最后,也欢迎来我的工从号彤哥读源码系统地学习源码&架构的知识。

5. 彤哥说netty系列之Java NIO核心组件之Channel的更多相关文章
- 6. 彤哥说netty系列之Java NIO核心组件之Buffer
--日拱一卒,不期而至! 你好,我是彤哥,本篇是netty系列的第六篇. 简介 上一章我们一起学习了Java NIO的核心组件Channel,它可以看作是实体与实体之间的连接,而且需要与Buffer交 ...
- 7. 彤哥说netty系列之Java NIO核心组件之Selector
--日拱一卒,不期而至! 你好,我是彤哥,本篇是netty系列的第七篇. 简介 上一章我们一起学习了Java NIO的核心组件Buffer,它通常跟Channel一起使用,但是它们在网络IO中又该如何 ...
- 4. 彤哥说netty系列之Java NIO实现群聊(自己跟自己聊上瘾了)
你好,我是彤哥,本篇是netty系列的第四篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 上一章我们一起学习了Java中的BIO/NIO/AIO的故事,本章将带着大家一起使 ...
- 3. 彤哥说netty系列之Java BIO NIO AIO进化史
你好,我是彤哥,本篇是netty系列的第三篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 上一章我们介绍了IO的五种模型,实际上Java只支持其中的三种,即BIO/NIO/ ...
- 1. 彤哥说netty系列之开篇(有个问卷调查)
你好,我是彤哥,本篇是netty系列的第一篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 本文主要讲述netty系列的整体规划,并调查一下大家喜欢的学习方式. 知识点 ne ...
- 2. 彤哥说netty系列之IO的五种模型
你好,我是彤哥,本篇是netty系列的第二篇. 欢迎来我的公从号彤哥读源码系统地学习源码&架构的知识. 简介 本文将介绍linux中的五种IO模型,同时也会介绍阻塞/非阻塞与同步/异步的区别. ...
- Java NIO 核心组件学习笔记
背景知识 同步.异步.阻塞.非阻塞 首先,这几个概念非常容易搞混淆,但NIO中又有涉及,所以总结一下[1]. 同步:API调用返回时调用者就知道操作的结果如何了(实际读取/写入了多少字节). 异步:相 ...
- 一文让你彻底理解 Java NIO 核心组件
背景知识 同步.异步.阻塞.非阻塞 首先,这几个概念非常容易搞混淆,但NIO中又有涉及,所以总结一下. 同步:API调用返回时调用者就知道操作的结果如何了(实际读取/写入了多少字节). 异步:相对于同 ...
- 一文理解 Java NIO 核心组件
同步.异步.阻塞.非阻塞 首先,这几个概念非常容易搞混淆,但NIO中又有涉及,所以总结一下[1]. 同步:API调用返回时调用者就知道操作的结果如何了(实际读取/写入了多少字节). 异步:相对于同步, ...
随机推荐
- JVM(9) 程序编译及代码优化
一.早期(编译器)优化 1.编译期 Java 语言的 “编译期” 其实是一段 “不确定” 的操作过程,因为它可能是指 一个前端编译器(其实叫 “编译器的前端” 更准确一些)把 *.java 文件转变成 ...
- 数据结构(三十四)最短路径(Dijkstra、Floyd)
一.最短路径的定义 在网图和非网图中,最短路径的含义是不同的.由于非网图没有边上的权值,所谓的最短路径,其实就是指两顶点之间经过的边数最少的路径:而对于网图来说,最短路径是指两顶点之间经过的边上权值之 ...
- 【Linux系列】Linux基础知识整理
Linux操作系统在服务器领域广泛的使用到,作为一个后台开发工程师很有必要了解Linux相关的知识. 本篇日志是我学习Linux过程中的简单记录和总结.本着"理论够用,实践为主"的 ...
- docker 和 FastDFS上传和下载文件
1,从仓库拉取镜像 sudo docker image pull delron/fastdfs 也可以解压已经打包好的镜像文件 $ sudo docker load -i 文件路径/fastdfs_d ...
- Unity2-投影方式
- [考试反思]0714/0716,NOIP模拟测试3/4
这几天时间比较紧啊(其实只是我效率有点低我在考虑要不要坐到后面去吹空调) 但是不管怎么说,考试反思还是要写的吧. 第三次考试反思没写总感觉缺了点什么,但是题都刷不完... 一进图论看他们刷题好快啊为什 ...
- NOIP模拟 39
考的嘛也不是. 伤心(怎么可能) T1稍想想组合数,然后牢记: 取模题随时取模,包括刚刚读入的数据 T2想到了基环树,然而不会打QAQ.. 非常简洁但非常大神的做法:随便断掉环上的一条边 利用“这条 ...
- js基础总结02--字符串操作
1.字符串中对单个字符位置的操作 indexOf(char); 从左往右查找,返回匹配到的第一个字符的位置,没有匹配则返回-1 lastiIndexOf(char); 从右往左查找,返回匹配到的第一个 ...
- 常用的webpack优化方法
1. 前言 关于webpack,相信现在的前端开发人员一定不会陌生,因为它已经成为前端开发人员必不可少的一项技能,它的官方介绍如下: webpack 是一个模块打包器.webpack的主要目标是将 J ...
- lucas定理(模板题题解)
题目很简单,很暴力,就是组合数,没有其他的. 但是直接暴力会炸wow 我们可以利用Lucas定理来分解字问题. Lucas定理:C(n,m)(mod p)=C(n%p,m%p)*C(n/p,m/p)( ...