IO(BIO)和NIO区别:
本质区别就是阻塞与非阻塞的问题。
阻塞概念:应用程序在获取网络数据的时候,如果网络很慢,程序一直处于等待的状态,直到传输完毕。
非阻塞概念:应用程序直接获取已经准备好的数据,中间过程无需等待。
IO为同步阻塞,NIO为同步非阻塞,NIO并未实现异步。jdk1.7之后升级NIO,支持异步非阻塞通讯(AIO)
同步和异步都是面向操作系统与应用程序对IO操作的层面上来区别。
同步:应用程序会直接参与IO的读写操作,并且应用程序会阻塞到某一个方法上,知道数据准备就绪;或者采用轮询的策略实时检查数据的就绪状态,如果就绪就获取数据。
异步:所有的IO读写操作都交由操作系统处理,与我们的应用程序没有直接关系,应用程序不用关系IO读写,当操作系统完成了IO读写操作后,会直接给应用程序发送通知,应用程序直接拿去数据即可。
同步是server服务端的执行方式,阻塞表示接收数据的方式、状态(io、nio)
NIO编程介绍
Buffer:缓冲区
可以将数据直接写入或读取到stream中。nio中锁有的数据都是通过缓冲区处理(读写).常用ByteBuffer.
java中各种buffer类型如下,,除了booleanBuffer
ByteBuffer
CharBuffer
shortBuffer
IntBuffer
LongBuffer
FloatBuffer
DoubleBuffer
Channel:管道、通道
Selector:选择器、多路复用器、
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);
// 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);
*/
// 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) + ",");
}
*/
}
}