NIO浅析

Posted xhlwjy

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了NIO浅析相关的知识,希望对你有一定的参考价值。

一: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
View Code

说明:上面的程序是解释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
View Code

 

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

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

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

非直接缓冲区:

技术图片

 

 直接缓冲区:

技术图片

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

 直接缓冲区的坏处:

  a.创建和销毁开销大

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

 

 

未完待续,明天更新

 

以上是关于NIO浅析的主要内容,如果未能解决你的问题,请参考以下文章

Java NIO:浅析I/O模型

Java网络编程和NIO详解4:浅析NIO包中的BufferChannel 和 Selector

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

Java NIO浅析

JDK——NIO系统调用浅析

Java NIO:浅析I/O模型