NIO-Channel
NIO-Channel
目录
NIO-概览
NIO-Buffer
NIO-Channel
NIO-Channel接口分析
NIO-SocketChannel源码分析
NIO-FileChannel源码分析
NIO-Selector源码分析
NIO-WindowsSelectorImpl源码分析
NIO-EPollSelectorIpml源码分析
前言
本来是想学习Netty的,但是Netty是一个NIO框架,因此在学习netty之前,还是先梳理一下NIO的知识。通过剖析源码理解NIO的设计原理。
本系列文章针对的是JDK1.8.0.161的源码。
什么是Channel
通道(Channel)是对原I/O包中的流的模拟。与文件设备I/O交互的所有数据都必须通过一个Channel对象。
上一节我们提到在NIO中使用缓冲区来存放指定基元的数据,我们可以通过Buffer来读写数据。
将数据写入到硬盘时,我们可以将字节数据写入到缓冲区中;若我们要从硬盘读取数据,则需要通过通道将数据写入到缓冲区,然后再从缓冲区读取数据。
通道类型
根据不同的使用方式,分为不同的通道。比如我们需要网络读写,就需要网络交互的通道。需要文件读写就需要文件交互的通道。
NIO实现了Sctp协议、TCP协议、UDP协议以及文件传输四种通道,同时还实现了Windows平台的异步Socket通道以及异步文件通道。
windows平台的异步I/O是通过重叠I/O和IOCP(I/O完成端口)实现的,想要了解windows异步I/O的知识可以看一下我另一篇文章《Windows内核原理-同步IO与异步IO》
类型 | 通道 |
---|---|
Sctp协议客户端 | SctpChannel |
Sctp协议多播客户端 | SctpMultiChannel |
Sctp协议服务端 | SctpServerChannel |
UDP协议 | DatagramChannel |
TCP协议同步I/O服务端 | ServerSocketChannel |
TCP协议同步I/O客户端 | ServerChannel |
文件读写 | FileChannel |
对于Windows平台的异步通道
类型 | 通道 |
---|---|
TCP协议异步I/O服务端 | WindowsAsynchronousServerSocketChannel |
TCP协议异步I/O客户端 | WindowsAsynchronousSocketChannel |
异步文件读写 | WindowsAsynchronousFileChannel |
另外NIO还实现了一个单向通讯管道(Pipe)的功能,通过引入
SourceChannel
和SinkChannel
实现,底层实际还是Socket通讯。
如何使用
在介绍不同的Channel的实现之前我们先介绍下Channel如何使用。
ServerSocketChannel
以TCP协议为例,我们进行网络收发的时候,首先需要创建一个ServerSocketChannel用于监听端口。
//创建一个服务端socket通道用于接收连接
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
//绑定监听地址
serverSocketChannel.socket().bind(new InetSocketAddress(8080));
//等待连接
SocketChannel socketChannel = serverSocketChannel.accept();
我们监听了8080端口。若没有连接时,线程会阻塞在
accept
。
当有收到新的连接创建时,会获取到SocketChannel,此时我们需要创建一个Buffer用来从Channel中读取数据。
ByteBuffer buf = ByteBuffer.allocate(1024);
//数据将写入到buffer中
int length = socketChannel.read(buf);
数据写入到我们的Buffer中,我们就需要将他们读出来
buf.flip(); //转化为可读模式
byte[] data = new byte[length];
buf.get(data);
将数据从Buffer写入到Channel时
buf.clear();
byte[] resp = {'O','K'};
buf.put(resp);
buf.flip();//转换为读模式
socketChannel.write(buf);
这里为了方便直接使用原来的Buffer。
SocketChannel
作为客户端我们需要创建一个SocketChannel。
SocketChannel.open();
client.connect(new InetSocketAddress("127.0.0.1", 6060));
发送HELLO
给服务端
ByteBuffer buffer = ByteBuffer.allocate(10);
byte[] data = {'H', 'E', 'L', 'L', 'O'};
buffer.put(data);
buffer.flip();//转换为读模式
client.write(buffer);
阻塞等待读取数据
buffer.clear();
client.read(buffer);
buffer.flip();//转换为读模式
处理完成,需要关闭释放连接
//关闭客户端输入流
client.socket().shutdownInput();
//关闭客户端输出流
client.socket().shutdownOutput();
//关闭客户端socket时会关闭客户端channel
client.socket().close();
//关闭客户端channel,会同时关闭输入和输出流。
client.close();
关闭输出流会发送FIN包,若输入流未关闭仍然可以继续接收数据,这就是TCP的半连接。若处理完最后需要确保channel关闭。
FileChannel
FileChannel只能被FileInputStream、FileOutputStream、RandomAccessFile创建
RandomAccessFile
使用RandomAccessFile
创建FileChannel
//第一个参数时文件名,第二个参数是读写方式
RandomAccessFile randomAccessFile = new RandomAccessFile("1.txt","rw");
FileChannel channel = randomAccessFile.getChannel();
FileInputStream
使用RandomAccessFile
创建FileChannel
FileInputStream inputStream = new FileInputStream("1.txt");
channel = inputStream.getChannel();
inputStream获取的FileChannel只能读
FileOutputStream
使用RandomAccessFile
创建FileChannel
FileOutputStream outputStream = new FileOutputStream("1.txt");
channel = outputStream.getChannel();
inputStream获取的FileChannel只能写
关闭FileChannel
关闭FileChannel的方法和关闭SocketChannel方法一样。
//关闭channel时会关闭文件
channel.close();
//关闭文件时会关闭channel
randomAccessFile.close();
//关闭文件流时会关闭channel
inputStream.close();
//关闭文件流时会关闭channel
inputStream.close();
总结
由于源码解析的篇幅较长,因此将channel源码单独分出来讲解。
相关文献
微信扫一扫二维码关注订阅号杰哥技术分享
出处:https://www.cnblogs.com/Jack-Blog/p/12015516.html
作者:杰哥很忙
本文使用「CC BY 4.0」创作共享协议。欢迎转载,请在明显位置给出出处及链接。
NIO-Channel的更多相关文章
- Java NIO Channel和Buffer
Java NIO Channel和Buffer @author ixenos Channel和Buffer的关系 1.NIO速度的提高来自于所使用的结构更接近于OS执行I/O的方式:通道和缓冲器: 2 ...
- Java NIO Channel之FileChannel [ 转载 ]
Java NIO Channel之FileChannel [ 转载 ] @author zachary.guo 对于文件 I/O,最强大之处在于异步 I/O(asynchronous I/O),它允许 ...
- Java NIO Channel通道
原文链接:http://tutorials.jenkov.com/java-nio/channels.html Java NIO Channel通道和流非常相似,主要有以下几点区别: 通道可以读也可以 ...
- (三:NIO系列) Java NIO Channel
出处: Java NIO Channel 1.1. Java NIO Channel的特点 和老的OIO相比,通道和NIO流(非阻塞IO)主要有以下几点区别: (1)OIO流一般来说是单向的(只能读或 ...
- [翻译] java NIO Channel
原文地址:http://tutorials.jenkov.com/java-nio/channels.html JAVA NIO channels和流的概念很像,下面是他们的一些区别: 你可以对cha ...
- Java NIO Channel to Channel Transfers通道传输接口
原文链接:http://tutorials.jenkov.com/java-nio/channel-to-channel-transfers.html 在Java NIO中如果一个channel是Fi ...
- NIO Channel和Buffer
Java NIO 由以下几个核心部分组成: Buffer Channel Selector 传统的IO操作面向数据流,意味着每次从流中读一个或多个字节,直至完成,数据没有被缓存在任何地方.NIO操作面 ...
- Java NIO Channel to Channel Transfers
In Java NIO you can transfer data directly from one channel to another, if one of the channels is a ...
- NIO Channel的学习笔记总结
摘自:http://blog.csdn.net/tsyj810883979/article/details/6876603 1.1 非阻塞模式 Java NIO非堵塞应用通常适用用在I/O读写等方 ...
- NIO Channel Socket套接字相关Channel
阻塞非阻塞: NIO中的Channel主要分为两大类:一类是FileChannel,另一类是SocketChannel.NIO提供的核心非阻塞特性主要针对SocketChannel类,全部socket ...
随机推荐
- Kettle(6.0) 参数方式连接数据库
数据库连接条件(视自己实际情况而定) 数据库: Oracle 主机名称(IP): localhost 或 127.0.0.1 数据库名称(SID):MYORCL 端口号: 1521 用户名: scot ...
- 【IOS开发学习—OC篇】
一.instancetype和id关键字的区别 1)instancetype表示方法的返回类型和调用方法的对象类型相同. 2)在Object-C引入instancetype之前,初始化方法的返回类型都 ...
- LeetCode刷题总结-数组篇(下)
本期讲O(n)类型问题,共14题.3道简单题,9道中等题,2道困难题.数组篇共归纳总结了50题,本篇是数组篇的最后一篇.其他三个篇章可参考: LeetCode刷题总结-数组篇(上),子数组问题(共17 ...
- ie浏览器兼容性的入门解决方案
IE浏览器的兼容性素来是令人头疼的问题,大名鼎鼎的FUCK-IE不是浪得虚名的. 这里使用的解决方案是HACK,具体原理就是针对不同的浏览器写不同的HTML.CSS样式,从而使各种浏览器达到一致的渲染 ...
- 谷歌Chrome浏览器无法安装插件的解决方法(本文干货!)
这个问题困扰了我很久,作为小白学习可能会用到谷歌插件,奈何谷歌也太变态,国内的环境无法正常登录谷歌账户.无法访问应用商店,而Chrome主版本号大于66的只能从Chrome应用商店下载并安装插件,各种 ...
- Python基本数据结构之文件操作
用word操作一个文件的流程如下: 1.找到文件,双击打开 2.读或修改 3.保存&关闭 用python操作文件也差不多: f=open(filename) # 打开文件 f.write(&q ...
- 1.基础篇之vue入门
为了建立高效团队,很多公司会采用全栈工程师,虽然利弊兼有,对于成本优先的创业团队,肯定是首选,特别是对.net生态圈,大部分都是小公司,就更加重要了.这里记录的是对vue的学习点滴,希望对你有所助力. ...
- python while指令
while指令 1.它后面可以带一个 Ture 表示一直是真的,这样程序会在while循环里无限进行下去 eg: while Ture: print("打印这个...") 2.它后 ...
- Java传参-基本数据类型和引用数据类型作为参数的区别(值传递)
java中的方法可以传递参数,参数的传递方法就是值传递. 参数有形参和实参,定义方法时写的参数叫形参,真正调用方法时,传递的参数叫实参. 调用方法时,会把实参传递给形参,方法内部其实是在使用形参. 所 ...
- mysql 不需要使用密码就可以登录
最近发现一个问题, 就是我等了mysql客户端可以不输入密码. 直接输入mysql -u root 回车 或者 输入一个错的密码,都可进入到下面的界面. 在Navicat不用输入密码, 或者数据错的密 ...