一: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.数据进入直接缓冲区后,后续写入磁盘等操作就完全由操作系统决定了,不受我们控制
未完待续,明天更新
