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调用返回时调用者就知道操作的结果如何了(实际读取/写入了多少字节). 异步:相对于同步, ...
随机推荐
- fenby C语言 P9
逻辑运算 真1 假0 &&与 真&&真为真 真&&假为假 假&&真为假 假&&假为假 ||或 真||真为真 真||假为真 ...
- Spring的几种初始化和销毁方法
一 指定初始化和销毁方法 通过@Bean指定init-method和destroy-method: @Bean(initMethod="init",destroyMethod=&q ...
- linux文本编辑器教学
linux常见服务 一. 文本编辑器 vi vim是vi增强版 vim需要安装 sudo apt-get -y install vim 1 vim的三种工作模式 1 编辑模式 命令模式=>编辑模 ...
- 【java基础】为什么重写toString()方法?
不得不说,有很多java初学者写java实体类的时候,并没有真正理解重写toString() 方法,可能是口头知道也可能是跟风随带添加toString() 方法,并没有真正理解其意义,如果真要被问起来 ...
- 水管局长数据加强版:lct,时光倒流,最小生成树,边化点
Description: SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到 ...
- [UWP]使用SpringAnimation创建有趣的动画
1. 什么是自然动画 最近用弹簧动画(SpringAnimation)做了两个番茄钟,关于弹簧动画官方文档已经介绍得够详细了,这篇文章就摘录一些官方文档核心内容. 在传统UI中,关键帧动画(KeyFr ...
- linux 命令 | 常用命令导图(0)
- 查看redis占用内存大小的方法
查看redis占用内存大小的方法 <pre>redis-cli auth 密码info</pre><pre># Memory used_memory:1349009 ...
- centos6的JDK安装
1. 通过如下命令查看当前操作系统是否存在JDK rpm -qa | grep java 如果出现以下内容说明你的操作系统存在jdk 2.那么依次通过如下命令进行删除它 rpm -e - -nodep ...
- powerdesigner连接Mysql进行反向工程并生成word文档图文教程
1 软件版本 windows7 64位 powerdesigner 15.1 Mysql 5.1.56 mysql-connector-odbc-3.51.30-winx64 对于mysql-conn ...