一、IO与NIO的区别:

  前提我们先说一说java IO: 

  Java中使用IO(输入输出)来读取和写入,读写设备上的数据、硬盘文件、内存、键盘......,根据数据的走向可分为输入流输出流,这个走向是以内存为基准的,即往内存中读数据是输入流,从内存中往外写是输出流

  根据处理的数据类型可分为字节流和字符流:

  1.字节流可以处理所有数据类型的数据,在java中以Stream结尾

  2.字符流处理文本数据,在java中以Reader和Writer结尾

我们来看个IO流的详解图:

  1.NIO的起源:

    java.NIO是jdk1.4是提出,jdk1.7进行二次改进的java新型IO类,其最大的特点就是它的非阻塞性。

  2.IO与NIO结构差异:

    普通的IO是面向流(Stream Oriented),而NIO则是面向缓冲区(Buffer Oriented)。IO流是单向的,直接面向字节流,通过InputStream、OutputStream来完成数据的输入输出。而NIO是双向的,通过建立通道Channel),然后将数据装在缓冲区Buffer)在通道上进行传输。针对不同类型的数据有不同的Buffer,根据数据类型不同(boolean 除外),提供了相应类型的缓冲区: ByteBuffer、CharBuffer、ShortBuffer、IntBuffer、LongBuffer、 FloatBuffer、DoubleBuffer

二、NIO的Buffer的数据存取:

  1.创建Buffer实例:

ByteBuffer buf = ByteBuffer.allocate(1024); //1024为capacity,通过allocate()方法可以获取一个缓冲区

  2.Buffer类属性的解析:

    Buffer类中有三个属性必须理解:capacity(容量)、limit(访问范围)、position(位置,表示缓冲区中正在操作数据的位置)。通过get(),put()方法进行数据的存取。

    通过flip()方法切换成读模式,clear()方法清空缓冲区。但是缓冲区中的数据依然存在,但是处于“被遗忘”状态,rewind()可重复读。

  3.Buffer的分类:

    Buffer分为直接缓冲区非直接缓冲区。非直接缓冲区 通过allocate()方法分配缓冲区,将缓冲区建立在JVM的内存中。直接缓冲区 通过allocateDirect()方法,将缓冲区建立在物理内存中。这样做可以提高IO效率,节省了copy的过程,直接缓冲区是物理内存映射文件,但是写入过程不受控制,读过程受GC影响!

三、通道的原理与获取:

  1.通道的原理:

    传统的javaIO是通过DMA的方式存取,这种方式需要CPU的权限。而通道Channel)自带处理器,不需要去访问CPU,所以在进行大量IO时效率更高一些。通道用于源节点与目标节点的连接。在 Java NIO 中负责缓冲区中数据的传输。Channel 本身不存储数据,因此需要配合缓冲区进行传输。

  2.通道的获取:

    通道的主要实现类 java.nio.channels.Channel 接口:FileChannel、SocketChannel、ServerSocketChannel、DatagramChannel。

    获取通道的方法:

      1)各IO有自己的获取方法、

      2)jdk1.7的NIO2,针对各个通道提供了静态方法open();

      3)jdk1.7的NIO2的Files工具类的newByteChannel();

四、通道的数据传输:

  1.分散(Scatter)与聚集(Gather):  

    1)分散读取Scattering Reads),将Channel中读取的数据分散到Buffer
    2)聚集写入Gathering Writes),将多个Buffer中的数据聚集到Channel中

  2.实例代码:

        RandomAccessFile raf1 = new RandomAccessFile("1.txt", "rw");
//1. 获取通道
FileChannel channel = raf1.getChannel();
//2. 分配指定大小的缓冲区
ByteBuffer buf1 = ByteBuffer.allocate(250);
ByteBuffer buf2 = ByteBuffer.allocate(500);
//3. 分散读取
ByteBuffer[] bufs = {buf1, buf2};
channel1.read(bufs);
for (ByteBuffer byteBuffer : bufs) {
byteBuffer.flip();
}
System.out.println(new String(bufs[0].array(), 0, bufs[0].limit()));
System.out.println("-----------------");
System.out.println(new String(bufs[1].array(), 0, bufs[1].limit()));
//4. 聚集写入
RandomAccessFile raf2 = new RandomAccessFile("2.txt", "rw");
FileChannel channel2 = raf2.getChannel();
channel2.write(bufs);

五、字符集 Charset:

  编码:字符串->字节数组。解码:字节数组->字符串

  1.查看Charset里都有哪些编码:

        Map<String, Charset> map = Charset.availableCharsets();
map.forEach((k,v)->{
System.out.println(k);//常见的UTF-8等等..
});

  2.缓冲区编解码:

        Charset c = Charset.forName("UTF-8");
CharsetEncoder e = cs1.newEncoder(); //获取编码器
CharsetDecoder d = cs1.newDecoder(); //获取解码器
CharBuffer buf = CharBuffer.allocate(1024);
buf.put("二狗子到此一游");
buf.flip();
ByteBuffer bBuf = e.encode(buf );//编码
bBuf.flip(); //解码
CharBuffer buf2= d.decode(bBuf);
System.out.println(buf2.toString());

六、阻塞与非阻塞:

  阻塞与非阻塞式的,相较于网络通信:
    1)阻塞:C端发送S端读写请求,S端考虑,阻塞,服务端不会做其他事情,解决方式多线程
    2)非阻塞:Selector(选择器),通道会注册到选择器上,Selector监控IO状况,只有C端准备就绪

Java关于NIO类的详解的更多相关文章

  1. JAVA中StringBuffer类常用方法详解

    String是不变类,用String修改字符串会新建一个String对象,如果频繁的修改,将会产生很多的String对象,开销很大.因此java提供了一个StringBuffer类,这个类在修改字符串 ...

  2. Java日期工具类DateUtils详解(转)

    jar包 appache下的 common-lang3 一. 对指定的日期新增年.月.周.日.小时.分钟.秒.毫秒 public static Date addDays(Date date, int ...

  3. 在java poi导入Excel通用工具类示例详解

    转: 在java poi导入Excel通用工具类示例详解 更新时间:2017年09月10日 14:21:36   作者:daochuwenziyao   我要评论   这篇文章主要给大家介绍了关于在j ...

  4. 《手把手教你》系列基础篇(九十七)-java+ selenium自动化测试-框架设计篇-Selenium方法的二次封装和页面基类(详解教程)

    1.简介 上一篇宏哥介绍了如何设计支持不同浏览器测试,宏哥的方法就是通过来切换配置文件设置的浏览器名称的值,来确定启动什么浏览器进行脚本测试.宏哥将这个叫做浏览器引擎类.这个类负责获取浏览器类型和启动 ...

  5. java中的io系统详解 - ilibaba的专栏 - 博客频道 - CSDN.NET

    java中的io系统详解 - ilibaba的专栏 - 博客频道 - CSDN.NET 亲,“社区之星”已经一周岁了!      社区福利快来领取免费参加MDCC大会机会哦    Tag功能介绍—我们 ...

  6. Java中的main()方法详解

    在Java中,main()方法是Java应用程序的入口方法,也就是说,程序在运行的时候,第一个执行的方法就是main()方法,这个方法和其他的方法有很大的不同,比如方法的名字必须是main,方法必须是 ...

  7. Java虚拟机之垃圾回收详解一

    Java虚拟机之垃圾回收详解一 Java技术和JVM(Java虚拟机) 一.Java技术概述: Java是一门编程语言,是一种计算平台,是SUN公司于1995年首次发布.它是Java程序的技术基础,这 ...

  8. 使用Java操作文本文件的方法详解

    使用Java操作文本文件的方法详解 摘要: 最初java是不支持对文本文件的处理的,为了弥补这个缺憾而引入了Reader和Writer两个类 最初java是不支持对文本文件的处理的,为了弥补这个缺憾而 ...

  9. Java面试题04-final关键字详解

    Java面试题04-final关键字详解 本篇博客将会讨论java中final关键字的含义,以及final用在什么地方,感觉看书总会有一些模糊,而且解释的不是很清楚,在此做个总结,以备准备面试的时候查 ...

随机推荐

  1. perf4j 监控请求 + traceId区分日志

    1. 场景 从request进入Controller到出去的时间, 可以统计接口访问的一些数据,如:平均处理时间.最大处理时间 2. 代码 2.1 mvc-servlet 定义切面和拦截器 <? ...

  2. 2013年未之wpf项目乱述

    不知识为何现已很少在网上发帖,貌似人生的方向已经看的七七八八.要么用心工作,要么自主创业.无论怎么样,对于现在的我来说都是一种淡定的选择.作为一个c#程序员,今年下半年开始使用wpf,更觉得wpf将来 ...

  3. 基于zxing的二维码(网格)扫描

    基于zxing的二维码(网格)扫描 前言:对于二维码扫描我们使用的是开源框架Zxing或者Zbar,这里使用基于zxing的二维码扫描,类似支付宝网格扫描, 二维码原理介绍: 二维码是用某种特定的几何 ...

  4. swoole 创建tcp服务器

    server.php <?php /** * 创建tcp服务器 * Date: 2019/1/15 */ $serv = new swoole_server('127.0.0.1', 9501) ...

  5. python 小词云

    # Author:Alex.wang# Date:2017.06.02# Version:3.6.0 import matplotlib.pyplot as pltfrom wordcloud imp ...

  6. 关于实现XX系统设计时所实现的质量属性战术

    可用性: 1)使用Try-catch对抛出的异常进行处理 2)使用Spring事务管理 易用性: 1)在类似删除相关选项时,弹出提示框,防止误操作 2)在不编辑基本信息时,对其进行折叠或者隐藏 3)提 ...

  7. 从Linux访问Windows共享目录

    今天装备用VMWare装台Linux服务器来学习Oracle数据库,由于安装包是下载到本地的Windows系统,需要拷贝到Linux虚机里面去.搞了半天.想到虚机网络设成桥接,然后访问.百度了一下.最 ...

  8. Sandworm Attack小结

    这个漏洞刚出来时就分析过,当时大致弄明白了原理,但对很多细节和原理还是一知半解.后来开始找工作……今天终于有时间来把欠的这部分功课补上. 这个漏洞网上的各种中英文分析已经很多了,因此这里我只根据自己的 ...

  9. Java字符串工具类

    import java.io.ByteArrayOutputStream;import java.io.UnsupportedEncodingException;import java.lang.re ...

  10. 贝叶斯网络(Bayesian network))简介(PRML第8.1节总结)概率图模型(Graphical models)

    转:http://www.cnblogs.com/Dzhouqi/p/3204353.html 部分图为手写,由于本人字很丑,望见谅,只是想把PRML书的一些部分总结出来,给有需要的人看,希望能帮到一 ...