Java nio 概述

Java nio 全称Java new io,是jdk1.4里提供的api。

Java nio 与 io最主要的区别

1、读写方式

Io是面向流的,只能在一个方向上移动,一个输入流(inputStream)一个输出流(outputStream),需要从流中读一个或多个字节,直到读取完所有的字节。
Nio是面向缓冲的,可以双向,无论是读写,数据都会被先放到一个缓冲区

2.阻塞与非阻塞

Io流是阻塞的,当一个线程调用read()或write()时,直到被read或write完成,在此期间,该线程处于等待阻塞模式,不干任何其他事情。
Nio 是非阻塞的,当一个线程从通道发出读/写的请求的时候,数据会被先放到缓冲区里,而不需要等待全部读写完成,在此期间,这个线程可以去做别的事情,通常在空闲时候会执行其他通道上的io操作,一个线程可以管理多个通道

 

 


 

Nio核心概念:

 

Buffer (缓冲区),Channel(通道),Selector(选择器)

 

Buffer

 

前面已经提到,Nio是面向缓冲区的,所有的读写都要先放到缓冲区。

 

所有的缓冲区类型都继承于抽象类Buffer,最常用的是Bytebuffer, java基础类基本上都有相应的buffer类来一一对应,类继承关系图如下:

Channel

通道和io的流类似,主要区别为:通道是双向的,可读可写,io流是单向的,只能读或写,而且操作通道不会直接从通道中写入或读取数据,都是由通道将数据放入缓冲区(buffer)中。
最常用的通道:
Filechannel 读取/写入 文件数据
Socketchannel TCP协议的socket 读写数据
Datagramchannel UDP协议读写数据
下面是客服端发送读或写的请求过程,


Filechannel 往文件中读取和写入数据的简单例子

读取文件内容:

  1. public static void main(String[] args) {  
  2.         FileInputStream fin = null;  
  3.         try {  
  4.             fin = new FileInputStream("c:\\nio.txt");  
  5.         } catch (FileNotFoundException e) {  
  6.             e.printStackTrace();  
  7.         }  
  8.         // 获取通道  
  9.         FileChannel fc = fin.getChannel();  
  10.   
  11.         // 创建缓冲区  
  12.         ByteBuffer buffer = ByteBuffer.allocate(1024);  
  13.   
  14.         // 读取数据到缓冲区  
  15.         try {  
  16.             fc.read(buffer);  
  17.         } catch (IOException e) {  
  18.             e.printStackTrace();  
  19.         }  
  20.   
  21.         buffer.flip();  
  22.   
  23.         while (buffer.remaining() > 0) {  
  24.             byte b = buffer.get();  
  25.             System.out.print(((char) b));  
  26.         }  
  27.         try {  
  28.             fin.close();  
  29.         } catch (IOException e) {  
  30.             e.printStackTrace();  
  31.         }  
  32.           
  33.     }  


写入文件内容:

  1. public static void main(String[] args) {  
  2.         File file = new File("c:\\nio.txt");   
  3.         FileOutputStream outputStream = null;  
  4.         try {  
  5.             outputStream = new FileOutputStream(file);  
  6.         } catch (FileNotFoundException e1) {  
  7.             e1.printStackTrace();  
  8.         }   
  9.         FileChannel channel = outputStream.getChannel();   
  10.         ByteBuffer bufferWrite = ByteBuffer.allocate(1024);   
  11.         String string = "hello nio...";   
  12.         bufferWrite.put(string.getBytes());   
  13.         //这里必须调用flip(),先读取数据到Buffer,再从Buffer中读取数据。  
  14.         bufferWrite.flip();  
  15.         try {  
  16.             channel.write(bufferWrite);  
  17.             channel.close();   
  18.             outputStream.close();   
  19.         } catch (IOException e1) {  
  20.             e1.printStackTrace();  
  21.         }   
  22.     }  


Select 
Select 能检测到注册的所有通道上是否有读写请求,当有请求的时候才会进行读写,一个线程管理了多个通道,避免了多线程切换导致的开销,也不用去维护多个线程,操作原理如下图

这里介绍了javaio和nio的区别,以及nio的主要概念,还有简单的nio读写文件数据的例子,对nio就不做深入了。

有兴趣的同学可以看Jakob Jenkov的系列文章,http://tutorials.jenkov.com/java-nio/index.html

接下来会分享nio框架netty的一些学习心得,以及netty在实际项目架构中的使用

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值