java 网络通信 NlO(non-blocking i/o 或者 new i/o) Buffer

  • NIO,异步非阻塞的概念是jdk1.7之后才有的,nio1.0只实现了非阻塞,并没有实现异步
  • 再谈select, iocp, epoll,kqueue及各种I/O复用机制
  • NIO里的三个概念:缓冲区(buffer),管道(channel),选择器(selector也叫多路复用器)
  • NIO buffer 使用,buffer 有三个重点属性,pos(位置),lim(上限),cap(初始化容量)
import java.nio.IntBuffer;

public class TestBuffer {
    
    public static void main(String[] args) {
        
        // 1 基本操作
        //创建指定长度的缓冲区
        IntBuffer buf = IntBuffer.allocate(10);
        buf.put(13);// position位置:0 - > 1
        buf.put(21);// position位置:1 - > 2
        buf.put(35);// position位置:2 - > 3
        //把位置复位为0,也就是position位置:3 - > 0
        buf.flip();
        System.out.println("使用flip复位:" + buf);
        System.out.println("容量为: " + buf.capacity());   //容量一旦初始化后不允许改变(warp方法包裹数组除外)
        System.out.println("限制为: " + buf.limit());      //由于只装载了三个元素,所以可读取或者操作的元素为3 则limit=3
        
        
        System.out.println("获取下标为1的元素:" + buf.get(1));
        System.out.println("get(index)方法,position位置不改变:" + buf);
        buf.put(1, 4);
        System.out.println("put(index, change)方法,position位置不变:" + buf);;
        
        for (int i = 0; i < buf.limit(); i++) {
            //调用get方法会使其缓冲区位置(position)向后递增一位
            System.out.print(buf.get() + "\t");
        }
        System.out.println("buf对象遍历之后为: " + buf);
        /**
         * 总结
         * put(value) 和get() 都会导致position位置向后移一位
         * get(index) 和 put(index,value) 带下标的方法不会导致position改变
         */

        // 2 wrap方法使用

        //  wrap方法会包裹一个数组: 一般这种用法不会先初始化缓存对象的长度,因为没有意义,最后还会被wrap所包裹的数组覆盖掉。 
        //  并且wrap方法修改缓冲区对象的时候,数组本身也会跟着发生变化。
        /*
        int[] arr = new int[]{1,2,5};
        IntBuffer buf1 = IntBuffer.wrap(arr);
        System.out.println(buf1);
        
        IntBuffer buf2 = IntBuffer.wrap(arr, 0 , 2);
        //这样使用表示容量为数组arr的长度,但是可操作的元素只有实际进入缓存区的元素长度
        System.out.println(buf2);
        */
        /**
         * 总结
         * wrap 方法包裹数组时,有些稍微不一样的地方,就像将值逐个put之后,flip了一样
         * 包裹数组之后返回的buffer的三个属性是这样的pos=0,lim=arr.length,cap=arr.length
         * 包裹数组可以添加从哪开始,包裹几个元素,也就是wrap(arr,offset,length)
         */
        

        // 3 其他方法
        IntBuffer buf1 = IntBuffer.allocate(10);
        int[] arr = new int[]{1,2,5};
        buf1.put(arr);
        System.out.println(buf1);
        //一种复制方法
        IntBuffer buf3 = buf1.duplicate();
        System.out.println(buf3);
        
        //设置buf1的位置属性
        //buf1.position(0);
        buf1.flip();
        System.out.println(buf1);
        
        System.out.println("可读数据为:" + buf1.remaining());
        
        int[] arr2 = new int[buf1.remaining()];
        //将缓冲区数据放入arr2数组中去
        buf1.get(arr2);
        for(int i : arr2){
            System.out.print(Integer.toString(i) + ",");
        }

        /**
         * 总结
         * 可以put 数组,此时buffer的属性如下
         * pos=arr.length,lim=cap,cap=cap
         * 复制一个buffer可以使用duplicate()方法,
         * pos的位置可以设置
         * 获取可读数据的方法是remaining(),返回是pos到lim的差值
         * get(arrary) 可以通过array存放buffer里的数据
         * position(pos) 方法不建议使用因为你设置的pos可能导致一些无用的数据跑出来没有意义,
         * 使用put(value) 和get()之后记得flip, 应为这俩方法都会导致position位置向后移一位
         */


    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • channel 是双向的,同时读取和写入,流是单向的 channel 和多路复用器结合之后,有多种状态位,方便多路...
    MicoCube阅读 616评论 0 0
  • 作者: 一字马胡 转载标志 【2017-11-24】 更新日志 一、Java OIO Java OIO (Jav...
    一字马胡阅读 1,404评论 0 12
  • 继上一篇文章《网络编程之IO与NIO阻塞分析》的讲解,已经知道了网络编程的基本方式,今天将继续进行网络编程相关概念...
    landy8530阅读 1,151评论 0 3
  • IOC是一种可以帮助我们解耦各业务对象间依赖关系的对象绑定方式。 一、注入方式(通常我们会说IOC是通过DI来实现...
    远o_O阅读 227评论 0 0
  • 三面朝阳的屋子,夏日的早上赖床都成了奢望,阳光穿透所有的窗格,刺入眼睛。六点晨起,屋里已挤满扎眼的光亮,像这样的日...
    贤儿玥儿阅读 463评论 5 9