NIO:新IO,同步的非阻塞IO。

1.Java NIO 由以下几个核心部分组成:Channels(通道)、Buffers(缓冲区)、Selectors(选择器)

Channels(通道)

1.所有的 IO 在NIO 中都从一个Channel 开始。

Channel用来读取和写入数据,类似于之前的输入流/输出流。

Channel通道是双向的,通过一个Channel既可以进行读,也可以进行写;而Stream只能进行单向操作,通过一个Stream只能进行读

一般不会直接操作通道,内容都是先读到或者写入到Buffers,再通过Buffers完成读写。

Buffer(缓冲区)

1.Buffer可以分为ByteBuffer、CharBuffer、ShortBuffer、IntBuffer、LongBuffer、FloatBuffer、DoubleBuffer

2.Buffer中有几个状态变量:

  • position: Buffer读取或写入的下一个操作指针,当向Buffer写入数据时此指针会改变,放在写入的最后一个元素之后。比如Buffer中写入了4个位置的数据,则position会指向第五个位置。
  • limit: 在写模式下,Buffer的limit表示你最多能往Buffer里写多少数据。写模式下,limit等于Buffer的capacity。
            当读模式时, limit表示你最多能读到多少数据。
  • capacity: 表示Buffer的最大容量,limit<=capacity。此值在分配Buffer时被设置,一般不会更改。

3.Buffer中的常用方法:

ByteBuffer.allocate(数量) : 开辟缓存区,初始化大小

flip()  :此方法将position设置为0,limit放到position位置。用于重设缓存区,以便接下来的输入输出。

get() :取出当前position的内容

put(): 在buffer里面写入内容

示例如下:

使用FileChannel配合Buffer读取文件,如下所示:

public class FileChannelRead {
public static void main(String[] args) throws IOException {
FileInputStream fileInputStream=new FileInputStream(new File("E:\\test2.txt"));
FileChannel fileChannel=fileInputStream.getChannel();
//初始化缓冲区大小
ByteBuffer byteBuffer=ByteBuffer.allocate(1024);
//将文件通道里面的字节读到缓冲区中
fileChannel.read(byteBuffer);
//此方法将position置为0,limit放到position位置
byteBuffer.flip();
// position<limit时返回true,用于判断是否还有数据。
while (byteBuffer.hasRemaining()) {
//读取缓冲区Buffer里面的内容
System.out.print((char)byteBuffer.get());
}
fileChannel.close();
}
}

使用FileChannel配合Buffer写入文件,如下所示:

public class FileChannelWrite {

    public static void main(String[] args) throws IOException {
FileOutputStream fileOutputStream=new FileOutputStream(new File("E:\\test2.txt"),true);
//获取输出的fileChannel(文件通道)
FileChannel fileChannel= fileOutputStream.getChannel();
//开辟缓冲
ByteBuffer byteBuffer=ByteBuffer.allocate(1024);
String name="lin";
//向缓冲区写入数据
byteBuffer.put(name.getBytes());
//重设缓冲区,准备输出
byteBuffer.flip();
//写入内容
fileChannel.write(byteBuffer);
fileChannel.close();
fileOutputStream.close();
System.out.println("写入成功");
}
}

Selector(选择器)

1.Selector允许单线程处理多个 Channel。如果你的应用打开了多个连接(通道),但每个连接的流量都很低,使用Selector就会很方便。

仅用单个线程来处理多个Channels的好处是,只需要更少的线程来处理通道。事实上,可以只用一个线程处理所有的通道。对于操作系统来说,线程之间上下文切换的开销很大,而且每个线程都要占用系统的一些资源(如内存)。

使用Selector配合ServerSocketChannel,进行网络操作,示例如下:

/**
* 创建一个非阻塞的服务器,向客户端返回当前的系统时间
*/
public class SelectorDemo {
public static void main(String[] args) throws IOException {
int ports[]= { 8000,8001,8002,8003,8004 } ;
Selector selector=Selector.open();
for(int port : ports) {
//打开服务器套接字通道
ServerSocketChannel serverSocketChannel=ServerSocketChannel.open();
//服务器配置设置为非阻塞
serverSocketChannel.configureBlocking(false);
ServerSocket serverSocket=serverSocketChannel.socket();
InetSocketAddress inetSocketAddress=new InetSocketAddress(port);
serverSocket.bind(inetSocketAddress);
//注册选择器,相当于使用accept()方法接收
serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT );
System.out.println("服务器运行,在"+port+"端口监听");
} int keysAdd;
//选择一组键,相当的通道已为IO准备就绪
while ( (keysAdd=selector.select())>0 ) {
//取出全部生成的key
Set<SelectionKey> selectionKeySet=selector.selectedKeys() ;
Iterator<SelectionKey> iterator=selectionKeySet.iterator();
//迭代全部的SelectionKey
while (iterator.hasNext()) {
SelectionKey key=(SelectionKey) iterator.next();
//判断客户端是否已经连接上
if(key.isAcceptable()) {
//通过key获取channel
ServerSocketChannel serverSocketChannel=(ServerSocketChannel) key.channel();
SocketChannel socketChannel=serverSocketChannel.accept();
serverSocketChannel.configureBlocking(false);
ByteBuffer byteBuffer=ByteBuffer.allocate(1024);
byteBuffer.put(( "当前时间为"+new Date() ).getBytes());
byteBuffer.flip();
socketChannel.write(byteBuffer);
socketChannel.close();
}
}
selectionKeySet.clear();
}
}
}

IO和NIO的区别:

  • IO是多线程的,阻塞的。NIO,是同步的非阻塞IO。
  • IO面向Stream(流),而NIO面向Buffer(缓冲区)
  • IO是多个线程的,不存在Selector。而Java NIO的Selector(选择器)允许一个单独的线程来监视多个Channel(输入通道)

代码示例见GitHub:

https://github.com/firefoxer1992/JavaDemo/tree/master/src/main/java/com/nio

参考资料:http://ifeve.com/java-nio-all/

NIO简单理解的更多相关文章

  1. git的简单理解及基础操作命令

    前端小白一枚,最近开始使用git,于是花了2天看了廖雪峰的git教程(偏实践,对于学习git的基础操作很有帮助哦),也在看<git版本控制管理>这本书(偏理论,内容完善,很不错),针对所学 ...

  2. 简单理解Struts2中拦截器与过滤器的区别及执行顺序

    简单理解Struts2中拦截器与过滤器的区别及执行顺序 当接收到一个httprequest , a) 当外部的httpservletrequest到来时 b) 初始到了servlet容器 传递给一个标 ...

  3. [转]简单理解Socket

    简单理解Socket 转自 http://www.cnblogs.com/dolphinX/p/3460545.html  题外话 前几天和朋友聊天,朋友问我怎么最近不写博客了,一个是因为最近在忙着公 ...

  4. Js 职责链模式 简单理解

    js 职责链模式 的简单理解.大叔的代码太高深了,不好理解. function Handler(s) { this.successor = s || null; this.handle = funct ...

  5. Deep learning:四十六(DropConnect简单理解)

    和maxout(maxout简单理解)一样,DropConnect也是在ICML2013上发表的,同样也是为了提高Deep Network的泛化能力的,两者都号称是对Dropout(Dropout简单 ...

  6. Deep learning:四十二(Denoise Autoencoder简单理解)

    前言: 当采用无监督的方法分层预训练深度网络的权值时,为了学习到较鲁棒的特征,可以在网络的可视层(即数据的输入层)引入随机噪声,这种方法称为Denoise Autoencoder(简称dAE),由Be ...

  7. 简单理解dropout

    dropout是CNN(卷积神经网络)中的一个trick,能防止过拟合. 关于dropout的详细内容,还是看论文原文好了: Hinton, G. E., et al. (2012). "I ...

  8. 我们为之奋斗过的C#-----C#的一个简单理解

    我们首先来简单叙述一下什么是.NET,以及C#的一个简单理解和他们俩的一个区别. 1 .NET概述 .NET是Microsoft.NET的简称,是基于Windows平台的一种技术.它包含了能在.NET ...

  9. 简单理解ECMAScript2015中的箭头函数新特性

    箭头函数(Arrow functions),是ECMAScript2015中新加的特性,它的产生,主要有以下两个原因:一是使得函数表达式(匿名函数)有更简洁的语法,二是它拥有词法作用域的this值,也 ...

随机推荐

  1. python中的split、rsplit、splitlines

    split()从左向右寻找,以某个元素为中心将左右分割成两个元素并放入列表中 rsplit()从右向左寻找,以某个元素为中心将左右分割成两个元素并放入列表中 splitlines()根据换行符(\n) ...

  2. 精通Web Analytics 2.0 (12) 第十章:针对潜在的网站分析陷阱的最佳解决方案

    精通Web Analytics 2.0 : 用户中心科学与在线统计艺术 第十章:针对潜在的网站分析陷阱的最佳解决方案 是时候去处理网站分析中最棘手的一些问题了,然后获得属于你的黑带,这是成为分析忍者的 ...

  3. Hessian 源码简单分析

    Hessian 是一个rpc框架, 我们需要先写一个服务端, 然后在客户端远程的调用它即可. 服务端: 服务端通常和spring 做集成. 首先写一个接口: public interface Hell ...

  4. HTML5 图片宽高自适应,居中裁剪不失真

    一,使用 JS,先上效果图,右图为定死宽高的效果,左图为处理之后的 1, 主要思路是,在图片 onload 后,将图片的宽高比和 div 容器的宽高比进行比较, 2, 从而确定拉伸或者压缩之后是宽还是 ...

  5. Linux 设置IP地址,并能连接外网

    1,如果是 centos6,请修改  vi /etc/sysconfig/network-scripts/ifcfg-eth0 2,如果是 centos7,请修改 => vi /etc/sysc ...

  6. 01.GOF设计模式_概述

    0.Abstract Fcatory 提供一个创建一系列相关或者相互依赖对象的接口,而无需指定它们具体的类. 1. Adapter 将一个类的接口转换成客户希望的另外一个接口.Adapter模式使得原 ...

  7. beego注解路由 [自定义方法]

    背景: beego生成的controller里面,默认get请求到由Get()方法处理:post请求由Post()方法处理 etc. 如果想自定义方法来处理请求,改怎么做? 直接拿beego的文档来说 ...

  8. day11-元组与字典

    1.元组Tuple与列表类似,不同之处在于元组的元素不能修改. 元组使用小括号,列表使用中括号.元组可以查询,可以使用内置函数count.index.但是不能修改.增加.删除(儿子不能,孙子有可能). ...

  9. idea安装下载

    https://blog.csdn.net/qq_41983010/article/details/82562975

  10. java.lang.ClassNotFoundException: org.hibernate.engine.SessionFactoryImplementor

    Hibernate4.x与spring3.x整合,有关事务的处理,用Junit4测试,出现org.springframework.beans.factory.BeanCreationException ...