NIO,一种基于通道和缓冲区的I/O方式,可以使用native函数库直接分配堆外内存,然后通过一个存储在java

堆的DirectBteBuffer对象作为这块内存的引用进行操作,避免了再java堆和native堆中来回复制数据。

NIO是一种同步非阻塞的IO模型。同步是指线程不断轮询IO事件是否就绪,非阻塞指线程在等待IO的时候,可以

同时做其他任务。同步的核心是Selector,Selector代替了线程本身轮询IO事件,避免了阻塞同时减少了不必要的

线程消耗;非阻塞的核心就是通道和缓冲区,当IO事件就绪时,可以通过写道缓冲区,保证IO的成功,而无需线程

阻塞式地等待。

Buffer

为什么NIO是基于缓冲区的IO方式,因为当一个连接建立完成后,IO数据未必会马上到达,为了当数据达到时能够正确

完成IO操作,在BIO(阻塞IO)中,等待IO的线程必须被阻塞,以全天候地进行IO操作。为了解决这种IO方式的低效问题,

引入了缓冲区的概念,当数据达到时,可以预先被写入缓冲区,再由缓冲区交给线程,因此线程无需阻塞等待IO

通道

当执行:SocketChannel.write(Buffer),将一个buffer写到一个通道中。通道来说相对比较抽象。通道是I/O传输发生时通过的入口

而缓冲区是这些数据传输的来源或目标。对于离开缓冲区的传输,你想传递出去的数据被置于一个缓冲区,被传送到通道。对于

传回缓冲区的传输,一个通道将数据放置于你所提供的缓冲区中。

例如有个服务器通道ServerSocketChannel serverChannel,一个客户端通道SocketChannel clientChannel;服务器缓存区

serverBuffer,客户端缓冲区clientBuffer。当服务器想向客户端发送数据时,需要调用clientChannel.write(serverBuffer)。当

客户端要读时,调用clientChannel.read(clientBuffer);当客户端向服务器发送数据时,需要调用serverChannel.write(clientBuffer)

当服务器要读时,调用serverChannel.read(serverBuffer); 可以理解为在NIO中,如果想将data发送到目标端,则需要将存储该

data的buffer,写入到目标端channel中,然后再从channel中读取数据到目标端的buffer中。

Selector

通道和缓冲区的机制,使得现场无需阻塞等待IO事件的就绪,但总的要有人来监管这些IO事件。 这就由selector来完成。selector允许

单个现场处理多个channel,如果你的应用打开了多个连接(通道),但每个连接的流量都很低,使用selector就比较方便

要使用selector,的向selector注册channel,然后调用它的select方法,这个方法会一直阻塞到某个注册的通道有事件就绪,这就是轮询。

一旦这个方法返回,线程就可以处理这些事件。

selector中注册的感兴趣事件有: OP_ACCEPT  OP_CONNECT  OP_READ  OP_WRITE

简单基于NIO方式实现server/client示例

package com.exe.learn.demo.nio;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator; public class ServerNio { private Selector selector; public void initServer(int port) throws IOException {
ServerSocketChannel serverChannel = ServerSocketChannel.open();
//设置非阻塞模式
serverChannel.configureBlocking(false);
serverChannel.socket().bind(new InetSocketAddress(port));
this.selector = Selector.open();
//将selector注册到服务端
serverChannel.register(selector, SelectionKey.OP_ACCEPT);
} public void listen() throws IOException {
System.out.println("服务端启动》》》》》");
while(true) {
selector.select();
Iterator<SelectionKey> ite = this.selector.selectedKeys().iterator();
while(ite.hasNext()) {
SelectionKey key = (SelectionKey) ite.next();
ite.remove();
if(key.isAcceptable()) {
ServerSocketChannel server = (ServerSocketChannel) key.channel();
SocketChannel socketChannel = server.accept();
socketChannel.configureBlocking(false);
socketChannel.write(ByteBuffer.wrap(new String("来自服务端的信息").getBytes()));
socketChannel.register(this.selector, SelectionKey.OP_READ);
}else if(key.isReadable()){
readFromClient(key);
}
}
}
} private void readFromClient(SelectionKey key) throws IOException {
SocketChannel server = (SocketChannel) key.channel();
ByteBuffer buffer = ByteBuffer.allocate(10);
server.read(buffer);
byte[] data = buffer.array();
String msg = new String(data).trim();
System.out.println("来自客户端的信息" + msg);
server.write(ByteBuffer.wrap(new String(msg).getBytes()));
} public static void main(String[] args) throws IOException {
ServerNio server = new ServerNio();
server.initServer(1024);
server.listen();
}
}
package com.exe.learn.demo.nio;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;
import java.util.Iterator; public class ClientNio { private Selector selector; public void initClient(String ip, int port) throws IOException {
SocketChannel socketChannel = SocketChannel.open();
socketChannel.configureBlocking(false);
this.selector = Selector.open();
//连接server端
socketChannel.connect(new InetSocketAddress(ip, port));
socketChannel.register(selector, SelectionKey.OP_CONNECT);
} public void listen() throws IOException {
System.out.println("客户端启动》》》》》》");
while(true) {
selector.select();
Iterator<SelectionKey> ite = this.selector.selectedKeys().iterator();
while(ite.hasNext()) {
SelectionKey key = ite.next();
ite.remove();
if(key.isConnectable()) {
SocketChannel socketChannel = (SocketChannel) key.channel();
if(socketChannel.isConnectionPending()) {
socketChannel.finishConnect();
}
socketChannel.configureBlocking(false);
socketChannel.write(ByteBuffer.wrap(new String("向服务器发送信息了".getBytes(), "UTF-8").getBytes()));
socketChannel.register(selector, SelectionKey.OP_READ);
}else if(key.isReadable()) {
readFromServer(key);
}
}
}
} private void readFromServer(SelectionKey key) throws IOException {
SocketChannel server = (SocketChannel) key.channel();
ByteBuffer buffer = ByteBuffer.allocate(10);
server.read(buffer);
byte[] data = buffer.array();
String msg = new String(data).trim();
System.out.println("来自服务端的信息" + msg);
server.write(ByteBuffer.wrap(new String(msg).getBytes()));
} public static void main(String[] args) throws IOException {
ClientNio client = new ClientNio();
client.initClient("127.0.0.1", 1024);
client.listen();
}
}

NIO摘录的更多相关文章

  1. NIO知识摘录

    在 JDK 1. 4 中 新 加入 了 NIO( New Input/ Output) 类, 引入了一种基于通道和缓冲区的 I/O 方式,它可以使用 Native 函数库直接分配堆外内存,然后通过一个 ...

  2. NIO学习笔记

    零.前言 这里整理摘录了我了解NIO的一些笔记. 参考资料: 1.深入浅出NIO之Channel.Buffer 2.深入浅出NIO之Selector实现原理 3.Java NIO vs. IO 一.N ...

  3. 转载 IO、文件、NIO【草案四】

    本章目录: 1.IO类相关内容 2.文件和目录 3.文件高级操作  NIO详解[1]——缓冲区(Buffer)[深入理解,总结自<Java-NIO>]: [*:下边的Buffer又指代抽象 ...

  4. tcp nio 远程主机强迫关闭了一个现有的连接

    import java.io.IOException; import java.net.InetSocketAddress; import java.net.ServerSocket; import ...

  5. 源码分析netty服务器创建过程vs java nio服务器创建

    1.Java NIO服务端创建 首先,我们通过一个时序图来看下如何创建一个NIO服务端并启动监听,接收多个客户端的连接,进行消息的异步读写. 示例代码(参考文献[2]): import java.io ...

  6. BIO\NIO\AIO记录

    IO操作可以分为3类:同步阻塞(BIO).同步非阻塞(NIO).异步(AIO). 同步阻塞(BIO):在此种方式下,用户线程发起一个IO操作以后,必须等待IO操作的完成,只有当真正完成了IO操作以后, ...

  7. MVC、MVP、MVVM、Angular.js、Knockout.js、Backbone.js、React.js、Ember.js、Avalon.js、Vue.js 概念摘录

    注:文章内容都是摘录性文字,自己阅读的一些笔记,方便日后查看. MVC MVC(Model-View-Controller),M 是指业务模型,V 是指用户界面,C 则是控制器,使用 MVC 的目的是 ...

  8. 支撑Java NIO 与 NodeJS的底层技术

    支撑Java NIO 与 NodeJS的底层技术 众所周知在近几个版本的Java中增加了一些对Java NIO.NIO2的支持,与此同时NodeJS技术栈中最为人称道的优势之一就是其高性能IO,那么我 ...

  9. Java I/O and NIO [reproduced]

    Java I/O and NIO.2---Five ways to maximize Java NIO and NIO.2---Build more responsive Java applicati ...

随机推荐

  1. leetcode_1293. Shortest Path in a Grid with Obstacles Elimination_[dp动态规划]

    题目链接 Given a m * n grid, where each cell is either 0 (empty) or 1 (obstacle). In one step, you can m ...

  2. [luogu]P1016 旅行家的预算[贪心]

    [luogu]P1016 旅行家的预算 题目描述 一个旅行家想驾驶汽车以最少的费用从一个城市到另一个城市(假设出发时油箱是空的).给定两个城市之间的距离D1.汽车油箱的容量C(以升为单位).每升汽油能 ...

  3. [USACO10MAR]伟大的奶牛聚集Great Cow Gat… ($dfs$,树的遍历)

    题目链接 Solution 辣鸡题...因为一个函数名看了我贼久. 思路很简单,可以先随便指定一个根,然后考虑换根的变化. 每一次把根从 \(x\) 换成 \(x\) 的一个子节点 \(y\),记录一 ...

  4. 【HDOJ6659】Acesrc and Good Numbers(dfs)

    题意:定义f(n,d)为数码d在1到n中出现的次数,其中d=0..9 如果f(d,k)=k,则称k是d好数 给定x和d,求不大于x的最大的d好数 x<=1e18 思路:考虑f的增长率主要和位数有 ...

  5. HTTP请求时候总是设置的两个参数ConnectionTimeOut和SocketTimeOut

    在HTTP请求时候总是设置两个参数,就是连接超时和Socket超时 public static final String SO_TIMEOUT = "http.socket.timeout& ...

  6. xxxxxxxxxxxxxxxxxxx

    <!DOCTYPE html> <html lang="en"> <head> <meta http-equiv="Conten ...

  7. jmeter测试https请求

    测试https请求时,需要添加证书 目录 1.下载证书 2.导入 3.执行https请求 1.下载证书 在浏览器中打开要测试的https协议的网站,以谷歌为例打开,下载证书到桌面 4.一直点击下一步 ...

  8. C# 做延迟,但系统又能同时能执行其它任务

    private void Delay(int Millisecond) //使用时直接调用即可 { DateTime current = DateTime.Now; while (current.Ad ...

  9. MySQL 查询语句--------------进阶8:分页查询

    #进阶8:分页查询 /* 应用场景:要显示的数据,一页显示不全,需要分页提交sql请求 语法: select 查询列表 from 表 [join type] join 表2 on 连接条件 [wher ...

  10. Cross-entropy Cost Function for Classification Problem

    在Machine Learning的Regression Problem中,常用Quadratic Function来做Cost Function,用以表征Hypothesis与Y之间的差距.而通过G ...