一:NIO与IO的区别

  1.NIO面对的是缓冲区,IO面对的是流

  2.NIO是非阻塞的,IO是阻塞的

  3.NIO中引入了选择器

二:既然NIO面对的是缓冲区,那就先来了解缓冲区

  1.NIO中Buffer负责存储,Buffer底层采用的是数组,可以存储不同数据类型,提供了相应的缓冲区(ByteBuffer,IntBuffer......),对于缓冲区的管理一致,通过allocate获取缓冲区

  2.缓冲区存取数据的2个方法,put()存入,get()取出

  3.缓冲区的4个核心属性

    a.capacity:容量,最大存储数据的容量,一旦声明不能改变

    b.limit:界限,表示缓冲区中可以操作数据的大小,(limit后面的数据不能进行读写)

    c.limit:界限,表示缓冲区中可以操作数据的大小,(limit后面的数据不能进行读写)

    d.position:位置,(表示缓冲区中正在操作数据的位置)

     e:规则:position<=limit<=capacity

    @Test
public void test1(){
String str="abcde";
//1.分配指定大小的缓冲区
ByteBuffer buf = ByteBuffer.allocate(1024);
System.out.println("allocatt.................................");
System.out.println(buf.position());
System.out.println(buf.limit());
System.out.println(buf.capacity()); //通过put存入缓冲区
buf.put(str.getBytes());
System.out.println("put.......................................");
System.out.println(buf.position());
System.out.println(buf.limit());
System.out.println(buf.capacity()); //切换成读数据模式
buf.flip();
System.out.println("flip.......................................");
System.out.println(buf.position());
System.out.println(buf.limit());
System.out.println(buf.capacity()); //读取缓冲区中的数据
byte[] dst=new byte[buf.limit()];
buf.get(dst);
System.out.println(new String(dst)); //获取完之后
System.out.println(buf.position());
System.out.println(buf.limit());
System.out.println(buf.capacity()); //rewide(),可重复读数据
System.out.println("rewinde.....................");
buf.rewind();
System.out.println(buf.position());
System.out.println(buf.limit());
System.out.println(buf.capacity()); //clear(),清空缓冲区,但是缓冲区中的数据依然存在,但是处于“被遗忘状态”(三个属性变为最初状态,不能正确的读取数据)
buf.clear();
System.out.println("clear.............................");
System.out.println(buf.position());
System.out.println(buf.limit());
System.out.println(buf.capacity());
}
allocatt.................................
0
1024
1024
put.......................................
5
1024
1024
flip.......................................
0
5
1024
abcde
5
5
1024
rewinde.....................
0
5
1024
clear.............................
0
1024
1024

说明:上面的程序是解释Buffer的基本属性,下面是其图解

    

  4.mark。标记,记录当前position的位置,通过reset恢复到mark位置
    @Test
public void test2(){
String str="abcde";
ByteBuffer buf = ByteBuffer.allocate(1024); buf.put(str.getBytes()); buf.flip();
byte[] dst=new byte[buf.limit()];
buf.get(dst,0,2);
System.out.println(new String(dst)); System.out.println(buf.position());
//标记一下
buf.mark(); buf.get(dst,2,2);
System.out.println(new String(dst));
System.out.println(buf.position()); //恢复
buf.reset();
System.out.println(buf.position());
}
ab
2
abcd
4
2

5.直接缓冲区与非直接缓冲区

* 非直接缓冲区:通过allocate()方法分配缓冲区,缓冲区建立再JVM内存中
* 直接缓冲区:通过allocateDirect()方法分配缓冲区,可以将缓冲区建立在操作系统的物理内存中.在某种情况下可以提高效率
    @Test
public void test3(){
//直接缓冲区
ByteBuffer buf = ByteBuffer.allocateDirect(1024);
}

图解直接缓冲区和非直接缓冲区:

非直接缓冲区:

直接缓冲区:

区别:直接缓冲区是直接在物理内存上面开辟空间,非直接缓冲区是在JVM上面开辟空间,在一定程度上面提高了效率

直接缓冲区的坏处:

  a.创建和销毁开销大

  b.数据进入直接缓冲区后,后续写入磁盘等操作就完全由操作系统决定了,不受我们控制

三:通道

  1.负责源节点和目标节点的连接,在NIO中负责缓冲区中数据的传输,不能存储数据,配合缓冲区使用(理解为铁路(通道)和火车(缓冲区))

  2.主要实现类

   java.nio.channels.Channel接口:
FileChannel--本地
SocketChannel--TCP
SertverSocketChannel-TCP
DatagramChannel-UDP 3.获取通道的三种方式*
  1.getChannel()方法
   
本地
    FileInputStream/FILEOut
    网络
Socket
ServerSocket
DatagramSocket
2.在JDK1.7中的NIO.2针对各个通道提供了静态方法open()
3.在JDK1.7中的NIO.2的Files的工具类的newByteChannel() 4.通道之间的数据传输
1.transferForm()
2.transferTo() 5.分散与聚集
1.分散读取(将通道中的数据分散到多个缓冲区)
2.聚集写入(将多个缓冲区中的数据聚集到通道中) 6.字符集
编码:将字符串转换成字符数组
解码:字符数组转换成字符串 下面举几个例子:
通过通道完成文件的复制
    @Test
public void test1() throws IOException {
FileInputStream fis=new FileInputStream("1.jpg");
FileOutputStream fos=new FileOutputStream("2.jpg"); //1.获取通道
FileChannel inChannel = fis.getChannel();
FileChannel outChannel = fos.getChannel(); //2.分配指定大小缓冲区
ByteBuffer buf=ByteBuffer.allocate(1024);
//3.将通道中的数据存入缓冲区
while(inChannel.read(buf)!=-1){
buf.flip();//切换到读取模式
//4.将缓冲区中的数据写入通道
outChannel.write(buf);
buf.clear();//清空缓冲区
}
//5.关闭通道
inChannel.close();
outChannel.close();
fis.close();
fos.close();
}

使用直接缓冲区完成文件的复制

    //使用直接缓冲区
@Test
public void test2() throws IOException {
FileChannel inChannel = FileChannel.open(Paths.get("1.jpg"), StandardOpenOption.READ);
FileChannel outChannel = FileChannel.open(Paths.get("3.jpg"), StandardOpenOption.WRITE,StandardOpenOption.READ,StandardOpenOption.CREATE); //内存映射文件
MappedByteBuffer inMapBuf = inChannel.map(FileChannel.MapMode.READ_ONLY, 0, inChannel.size());
MappedByteBuffer outMapBuf = outChannel.map(FileChannel.MapMode.READ_WRITE, 0, inChannel.size()); //直接堆缓冲区堆数据进行读写操作
byte[] dst=new byte[inMapBuf.limit()];
inMapBuf.get(dst);
outMapBuf.put(dst); inChannel.close();
outChannel.close();
}

通道之间的数据传输(使用直接缓冲区)

    //通道之间的数据传输(直接缓冲区)
@Test
public void test3() throws IOException {
FileChannel inChannel = FileChannel.open(Paths.get("1.jpg"), StandardOpenOption.READ);
FileChannel outChannel = FileChannel.open(Paths.get("3.jpg"), StandardOpenOption.WRITE,StandardOpenOption.READ,StandardOpenOption.CREATE); inChannel.transferTo(0,inChannel.size(),outChannel);
inChannel.close();
outChannel.close();
}

分散与聚集

说明:分散读取是将通道里面的数据依次读到多个缓冲区里面,聚集写入是将多个缓冲区里面的数据依次写入通道里面

    //分散与聚集
@Test
public void test4() throws IOException {
RandomAccessFile raf1=new RandomAccessFile("1.txt","rw"); //获取通道
FileChannel channel = raf1.getChannel(); //分配指定大小的缓冲区
ByteBuffer bf1=ByteBuffer.allocate(100);
ByteBuffer bf2=ByteBuffer.allocate(1024); //分散读取
ByteBuffer[] bufs={bf1,bf2};
channel.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())); //聚集写入
RandomAccessFile raf2=new RandomAccessFile("2.txt","rw");
FileChannel channel2 = raf2.getChannel();
channel2.write(bufs);
}

console:

abvc
dse
fds
gfdgfd gfdg
gfdg
gfdghytj
fgdgfcv
gfdr
xzcas
c
fds
aaaaaaaaaaaaaaaaaaaaaa
--------------------------------------------------
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa

字符集

    @Test
public void test5(){
SortedMap<String, Charset> map = Charset.availableCharsets(); Set<Map.Entry<String, Charset>> set = map.entrySet(); for (Map.Entry<String, Charset> entry:set) {
System.out.println(entry.getKey()+"="+entry.getValue());
}
}

编码和解码

****使用哪种字符集编码就要使用哪种字符集解码

    @Test
public void test6() throws CharacterCodingException {
Charset cs1 = Charset.forName("GBK");
//获取编码器和解码器
CharsetEncoder ce = cs1.newEncoder();
//获取解码器
CharsetDecoder cd = cs1.newDecoder(); CharBuffer cbuf = CharBuffer.allocate(1024);
cbuf.put("xhww!!");
cbuf.flip(); //编码
ByteBuffer bBuf = ce.encode(cbuf);
for (int i=0;i<12;i++){
System.out.println(bBuf.get());
}
//解码
bBuf.flip();
CharBuffer cBuf = cd.decode(bBuf);
System.out.println(cBuf);
}

NIO浅析(一)的更多相关文章

  1. Java NIO浅析 转至 美团技术团队

    出处: Java NIO浅析 NIO(Non-blocking I/O,在Java领域,也称为New I/O),是一种同步非阻塞的I/O模型,也是I/O多路复用的基础,已经被越来越多地应用到大型应用服 ...

  2. Java NIO浅析

    NIO(Non-blocking I/O,在Java领域,也称为New I/O),是一种同步非阻塞的I/O模型,也是I/O多路复用的基础,已经被越来越多地应用到大型应用服务器,成为解决高并发与大量连接 ...

  3. NIO浅析(二)

    一:前言 在(一中了解了NIO中的缓冲区和通道),通过本文章你会了解阻塞和非阻塞,选择器,管道 二:完成NIO通信的三要素 * 1.通道(Channel):负责连接* java.nio.channel ...

  4. 浅析I/O模型及其设计模式

    前言 I/O在软件开发中的重要性无需多言,无论是在操作系统.网络协议.DBMS这种底层支撑软件还是在移动APP,大型网站服务器等应用软件的开发中都是最核心最重要的部分.特别是现在软件服务使用量和数据量 ...

  5. NIO (一) NIO是什么

    参考文档:java为什么需要NIO:https://liuchi.coding.me/2017/08/01/浅谈Java为什么需要NIO/美团技术团队 NIO浅析:https://tech.meitu ...

  6. Java IO NIO详细讲解

    1.IO Java IO概述 2.NIO Java NIO浅析

  7. NIO教程 ——检视阅读

    NIO教程 --检视阅读 参考 BIO,NIO,AIO 总结 Java NIO浅析 Java NIO 教程--极客,蓝本 Java NIO 系列教程 --并发编程网 BIO,NIO--知乎 NIO 入 ...

  8. 也谈Reactor模式

    何谓Reactor模式?它是实现高性能IO的一种设计模式.网上资料有很多,有些写的也很好,但大多不知其所以然.这里博主按自己的思路简单介绍下,有不对的地方敬请指正. BIO Java1.4(2002年 ...

  9. 响应式编程系列(一):什么是响应式编程?reactor入门

    响应式编程 系列文章目录 (一)什么是响应式编程?reactor入门 (二)Flux入门学习:流的概念,特性和基本操作 (三)Flux深入学习:流的高级特性和进阶用法 (四)reactor-core响 ...

随机推荐

  1. Linux操作系统(五)_部署Zentao

    一.部署Zentao 1.检查服务器信息 uname -a 2.下载相应的部署包(一键安装包) http://dl.cnezsoft.com/zentao/9.8.1/ZenTaoPMS.9.8.1. ...

  2. shell脚本对代码执行时间的计时

    时间秒: $(date +%s) 算数表达式: $(($cost_time/60)) $(($cost_time%60)) #!/bin/bash # ccache is from epel mirr ...

  3. jQuery对于demo元素的上移、下移、删除操作等实现

    今天给大家分享一个实用的jQuery技能:dom元素的操作:我们经常会去获取dom元素去实现交互效果,以及数据的操作. 首先复习一下jQuery DOM 元素方法: .get() 获得由选择器指定的D ...

  4. JQ的live学习

    $("#StartTime").live("blur keypress keyup",function(){ if($("#EndTime" ...

  5. CentOS使用手册(三)

    前言: 目录:暂时不写,因为有些实验,比如负载均衡,反向代理,配置ssl等实验来不及做.所以这篇随笔还需日后补充(排版以后慢慢调,现在该做平台分析系统了) Linux中Mongodb4.x安装调试.远 ...

  6. 简单递归____Fibonacci数列

    #include <stdio.h> int fun(int x) { ||x==) ; else return fun(x-1)+fun(x-2); } int main() { int ...

  7. Android开发中怎样用多进程、用多进程的好处、多进程的缺陷、解决方法(转)

    转自:http://blog.csdn.net/spencer_hale/article/details/54968092 1.怎样用多进程 Android多进程概念:一般情况下,一个应用程序就是一个 ...

  8. OAuth_2

    角色: OAuth2.0为用户和应用定义了如下角色: 资源拥有者.资源服务器.客户端应用.授权服务器 资源拥有者:拥有共享数据的人或应用,比如Facebook的用户就是 资源拥有者,但资源拥有者也可以 ...

  9. 77th ploty 制图神器

    https://blog.csdn.net/u012897374/article/details/77857980

  10. windows10 配置SSH连接Github、配置SSH Key

    由于你的本地Git仓库和GitHub仓库之间的传输是通过SSH加密的,所以,需要设置SSH key. 第1步:创建SSH Key.在用户主目录下[我的电脑是C:\Users\ad],看看有没有.ssh ...