一、JAVA NIO概念

开始有的,之前只用过IO流,其实NIO和IO一样都是可以用来读取或者写入文件,只不过原来的IO是面向流进行操作的,而NIO是面向缓冲区进行操作

二、通过一个小例子初步了解下NIO如何进行文件读写




java 写入手机文件 java nio写入文件_System

java 写入手机文件 java nio写入文件_java_02

package com.boke.nio.base;

import java.io.File;
import java.io.FileInputStream;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;

/**
 * 原IO和NIO分别实现读取、写入文件
 * @author lenovo
 *
 */
public class MyFirstNio {
    
    public static void main(String[] args) throws Exception {
        File file = new File("C:/Users/lenovo/Desktop/test.txt");
        readio(file);//原IO
        readnio(file);//NIO
    }
    
    
    //原IO进行文件读操作
    public static void readio(File file) throws Exception{
        FileInputStream fis = new FileInputStream(file);
        byte[] arr = new byte[1024];
        int len = -1;
        System.out.println("原IO:");
        while((len = fis.read(arr))!=-1){
            System.out.println(new String(arr,0,len,"utf-8"));
        }
        
        if(fis!=null){
            fis.close();
        }
    }
    
    //NIO方式进行文件读操作
    public static void readnio(File file) throws Exception{
        FileInputStream fis = new FileInputStream(file);
        FileChannel channel = fis.getChannel();//通过流获取唯一通道
        ByteBuffer buffer  = ByteBuffer.allocate(1024);//创建一个缓冲区
        
        System.out.println("NIO:");
        while (true) {  
            // clear方法重设缓冲区,使它可以接受读入的数据  
            buffer.clear();  
  
            // 从输入通道中将数据读到缓冲区  
            int r = channel.read(buffer);  
  
            // read方法返回读取的字节数,可能为零,如果该通道已到达流的末尾,则返回-1  
            if (r == -1) {  
                break;  
            }  
            
            byte[] array = buffer.array();
            System.out.println(new String(array,0,r,"utf-8"));
        }  
        
        if(fis!=null){
            fis.close();
        }
        
    }
}


View Code


  通过上面的例子,可以很明显的看出IO和NIO一个最大的区别,IO是直接将数据读取至流中,而NIO数据终将在缓冲区中

三、根据例子对NIO关键概念进行讲解

  (1)例子中的FileChannel 

    通道,可以理解为它就是原IO中的流,但是读取的数据不是放在流中,而必须通过channel.read(buffer)将数据读取到缓冲区buffer中,如果取数据的话也是先将数据放至缓冲区中。

     通道与流的不同之处在于通道是双向的。而流只是在一个方向上移动(一个流必须是InputStream或者OutputStream的子类), 而通道可以用于读、写或者同时用于读写。因为它们是双向的,所以通道可以比流更好地反映底层操作系统的真实情况。特别是在UNIX模型中,底层操作系统通道是双向的。

  (2)例子中的ByteBuffer

     说明一下,ByteBuffer并不是NIO中唯一的缓冲区类型,是使用最多的一种缓冲区类型,当然还有其他缓冲区对象,只是类型不同而已,包括:

       ByteBuffer 
       CharBuffer 
       ShortBuffer 
       IntBuffer 
       LongBuffer 
       FloatBuffer 
       DoubleBuffer 

    缓冲区实质上是一个数组。通常它是一个字节数组,但是也可以使用其他种类的数组。但是一个缓冲区不仅仅是一个数组。每一个缓冲区都有复杂的内部统计机制,接下来我会开始了解一下。