在Java NIO中用Channel来对程序与进行I/O操作主体的连接关系进行抽象,这些IO主体包括如文件、Socket或其他设备。简而言之,指代了一种与IO操作对象间的连接关系。

  按照Channel接口的定义,Channel只有open和closed两种状态,只有在channel处于open状态下对其操作时才有效,而对closed的channel进行操作会导致抛出异常。相应的Channel接口也仅有isOpen()和close()两种方法。

  在Socket编程中,常用的Channel类是ServerSocketChannel和SocketChannel。ServerSocketChannel表示作为Socket服务端,处于监听状态的Socket连接。而SocketChannel表示一个已经建立的Socket连接。

  ServerSocketChannel具有accept()方法,当该方法成功返回时,会返回一个已经建立的SocketChannle对象,这也表示了这两种代表不同状态的Socket Channel间的联系和区别。ServerSocketChannel通过bind()方法绑定到本地地址,并且可以通过setOption()方法设置SO_RCVBUF(Socket接收缓冲区大小)和SO_REUSEADDR(是否重用Socket地址,由于Socket在关闭时需要经过多个状态的变迁,这个选项通常在需要快速重启的场景下有用)两个Socket选项。

  SocketChannel采用connet()方法建立连接,并且支持非阻塞式建立,在非阻塞状态下连接的最终建立状态通过isConnetionPending()和finishConnect()进行组合判断。SocketChannel同时支持异步关闭,若写线程主动关闭了Channel,则并发的读操作将读取不到内容;若读线程主动关闭了Channel,则并发的写操作将得到AsynchronousCloseException。同样的对SocketChannle可以通过setOption()方法设置若干选项,包括:SO_SNDBUF(发送缓冲区), SO_RCVBUF, SO_KEEPALIVE(是否采用协议栈实现的保活机制,默认为FALSE), SO_REUSEADDR, SO_LINGER(阻塞式连接下的关闭等待时长), TCP_NODELAY(是否禁用Nagle算法)。

  这两个类涉及的相关类图如下图所示:

  可见,ServerSocketChannel和SocketChannel均继承了SelectableChannel以支持与Selector配合实现非阻塞式IO。SocketChannel继承了若干支持读写操作的Channel以支持读写操作。

  另外,Channel的读写需要借助于Buffer,可见在NIO中Buffer是程序中其他对象与Channel交互的中介,通过Buffer和Channel的配合实现对I/O对象的读写操作。

  几个关键的接口定义如下

  

public interface Channel {

	/**
* 判断是否开启
* @return
*/
public boolean isOpen(); public void close() throws IOException; } /**
* 可中断Channel</p>
* 1. 可被异步关闭. 若当前线程阻塞于Channel的读写操作,</l>
* 当另一线程调用close操作后,当前线程会收到一个AsynchronousCloseException</p>
*
* 2. 可中断. 若当前线程阻塞于Channel,当其他线程触发interrupt方法时</l>
* 会导致Channel Close,当前线程收到一个ClosedByInterruptException,
* 其interrupt状态被设置</p>
*
* 3. 若当前线程interrupt状态已被设置,当调用当前Channel的阻塞IO操作时,</l>
* channel会被close,线程会收到ClosedByInterruptException. </l>
* 其interrupt状态保持不变.</p>
*
* @author luojiahu
*
*/
public interface InterruptibleChannel extends Channel{ public void close() throws IOException; } /**
* 可读Byte Channel
* 可对当前Channel阻塞读. 若有其他线程正对当前Channel进行读操作,</l>
* 调用read会一直阻塞至其他上一读操作完成。
*
* @author luojiahu
*
*/
public interface ReadableByteChannel extends Channel { /**
* 从当前Channel中读取内容写入Buffer</l>
* 是否能够读入内容,取决于当前Channel的状态.</l>
* 如,对于一个非阻塞的Socket Channel,如果当前socket的</l>
* 接收缓冲区没有任何内容,则不能写入任何内容到src
* @param src
* @return
* @throws IOException
*/
public int read(ByteBuffer src) throws IOException; } /**
* 可写Byte Channel
* 可对当前Channel阻塞写.</l>
* @author luojiahu
*
*/
public interface WritableByteChannel extends Channel { /**
* 从dst buffer中读取内容写入当前Channel</l>
* 是否能够写入内容取决于当前Channel的状态</l>
* 如对于非阻塞 Socket Channel,只能写入不大于</l>
* 对应写入缓冲区可写大小的内容
*
* @param dst
* @throws IOException
*/
public void write(ByteBuffer dst) throws IOException; } /**
* 网络socket channel</l>
* @author luojiahu
*
*/
public interface NetWorkChannel extends Channel{ /**
* 绑定到本地地址</l>
* 一旦绑定,则绑定直至通道关闭</l>
* 如果参数未空,则绑定至自动分配的地址
* @param local
* @return
* @throws IOException
*/
NetWorkChannel bind(SocketAddress local) throws IOException; /**
* 获取绑定的本地地址
* @return
* @throws IOException
*/
SocketAddress getLocalAddress() throws IOException; /**
* 设定socket 选项
* @param name
* @param value
* @return
* @throws IOException
*/
<T> NetworkChannel setOption(SocketOption<T> name, T value) throws IOException; }

  

Java NIO 之 Socket Channel的更多相关文章

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

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

  2. Java NIO学习笔记---Channel

    Java NIO 的核心组成部分: 1.Channels 2.Buffers 3.Selectors 我们首先来学习Channels(java.nio.channels): 通道 1)通道基础 通道( ...

  3. Java NIO之通道Channel

    channel与流的区别: 流基于字节,且读写为单向的. 通道基于快Buffer,可以异步读写.除了FileChannel之外都是双向的. channel的主要实现: FileChannel Data ...

  4. Java NIO中的Channel接口

    1. Channel  通道,可以将指定文件的部分或全部直接映射成Buffer. 不能直接读写Channel中的数据,Channel只能与ByteBuffer交互. 读数据时,把Channel中的数据 ...

  5. JAVA NIO 之Channel

    缓冲区本质上是一块可以写入数据,然后可以从中读取数据的内存.Channel 通道就是将数据传输给 ByteBuffer 对象或者从 ByteBuffer 对象获取数据进行传输. Channel 用于在 ...

  6. Java NIO学习系列二:Channel

    上文总结了Java NIO中的Buffer相关知识点,本文中我们来总结一下它的好兄弟:Channel.上文有说到,Java NIO中的Buffer一般和Channel配对使用,NIO中的所有IO都起始 ...

  7. Java NIO Channel 使用

    Java NIO 中的 Channel 分类: FileChannel SocketChannel ServerSocketChannel DatagramChannel channel 分类 Fil ...

  8. Java NIO 核心组件学习笔记

    背景知识 同步.异步.阻塞.非阻塞 首先,这几个概念非常容易搞混淆,但NIO中又有涉及,所以总结一下[1]. 同步:API调用返回时调用者就知道操作的结果如何了(实际读取/写入了多少字节). 异步:相 ...

  9. 3、谈谈 Java NIO

    在 JDK1.4 之后,为了提高 Java IO 的效率,Java 提供了一套 New IO (NIO),之所以称之为 New,原因在于它相对于之前的 IO 类库是新增的.此外,旧的 IO 类库提供的 ...

随机推荐

  1. 关于JS中变量提升的规则和原理的一点理解

        关于变量提升,以前在一些教程和书籍上都听到过,平时开发中也知道有这个规律,但是今天突然在一个公开课中听到时,第一反应时一脸懵逼,然后一百度,瞬间觉得好熟悉啊,差点被这个概念给唬住了,不信我给你 ...

  2. 前端基于react,后端基于.net core2.0的开发之路(番外篇) 后端使用T4模板,生成某些类

    1.介绍 因为开发过程中,有部分类是你加一个模型,就需要去改动的,每次加非常的烦,或者有些类,你只用到了他基类的方法,但是你还必须建一个文件才能调用他基类的方法,也很烦. 这个时候,T4就非常有用了. ...

  3. 更换HomeBrew源

    比较少用brew,只有之前安装Opencv的时候用过一次,后面有人问我怎么装,于是帮他研究了一下.MacOS的brew其实就是通过两个git仓库(brew和homebrew-core)来实现的源更新机 ...

  4. hihoCoder 1051 补提交卡(贪心,枚举)

    #1051 : 补提交卡 时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho给自己定了一个宏伟的目标:连续100天每天坚持在hihoCoder上提交一个程序.100天过 ...

  5. [bzoj 2017] [Usaco2009 Nov]硬币游戏

    一个多月没更博客了..(期间明白了自己有多傻逼. 这种问题大概就倒着做... f[i][j]:表示考虑剩下的硬币i..n,且之前的人取了j个时,先手最多拿到的钱数.aft[i]:表示硬币i..n的总钱 ...

  6. tju_4147 kd树+最小生成树

    kd树模板+全图最小生成树 标签(空格分隔): kd树+最小生成树 题目链接 题意: k维太空中有n个点,每个点可以与距离它m近的点连边,现在给你一堆点,并给出坐标,现在要建立通信网络,一些可以互相到 ...

  7. oracle游标的使用

    --游标的使用 步骤1.声明游标 .打开游标 .获取数据 .关闭游标 declare cursor cur is select * from emp; t_sal cur%rowtype; begin ...

  8. angular 表达式与指令

    angular表达式的一些特点 属性表达式: 属性表达式是对应于当前作用域,Javascript对应的是全局window对象. AngularJS要使用window作用域的话得用$window来指向全 ...

  9. seaJS 模块加载过程分析

    先看一个seajs的官方example,  以下以seajs.use('main')为例, 解析加载mod main的过程 //app.html seajs.use("main") ...

  10. textarea自适应高度,div模仿textarea可编辑实现自适应高度,placeholder使用图标

    1.textarea自适应高度,placeholder使用图标 自适应高度,有很多种办法: 1)jq: $("textarea").on("input",fun ...