NIO对字节流的读写

在执行IO时,Java的InputStream被广泛使用,比如DataInputStream.readInt等等。事实上,这些高度封装的接口奇慢无比。我有一个项目启动时需要读取90MB左右的词典文件,用DataInputStream耗时3秒以上,换用java.nio包直接操作内存字节,可以加速到300ms左右,整整提速10倍!当然,前提是你熟悉位运算。

java.nio中提供了两个类 FileChannel 和 ByteBuffer来将文件映射到内存,其中FileChannel表示文件通道,ByteBuffer是一个缓冲区。

具体步骤

①从FileInputStream、FileOutputStream以及RandomAccessFile对象获取文件通道

②将文件内存映射到ByteBuffer

③通过byteBuffer.array()接口得到一个byte数组

④直接操作字节

示例代码

  
  
  
  
  1.             FileInputStream fis = new FileInputStream(path);
  2.             // 1.从FileInputStream对象获取文件通道FileChannel
  3.             FileChannel channel = fis.getChannel();
  4.             int fileSize = (int) channel.size();
  5.  
  6.             // 2.从通道读取文件内容
  7.             ByteBuffer byteBuffer = ByteBuffer.allocate(fileSize);
  8.  
  9.             // channel.read(ByteBuffer) 方法就类似于 inputstream.read(byte)
  10.             // 每次read都将读取 allocate 个字节到ByteBuffer
  11.             channel.read(byteBuffer);
  12.             // 注意先调用flip方法反转Buffer,再从Buffer读取数据
  13.             byteBuffer.flip();
  14.             // 可以将当前Buffer包含的字节数组全部读取出来
  15.             byte[] bytes = byteBuffer.array();
  16.             byteBuffer.clear();
  17.             // 关闭通道和文件流
  18.             channel.close();
  19.             fis.close();
  20.  
  21.             int index = 0;
  22.             size = Utility.bytesHighFirstToInt(bytes, index);
  23.             index += 4;

其中,如果你当初使用了DataOutputStream.writeInt来保存文件的话,那么在读取的时候就要注意了。writeInt写入四个字节,其中高位在前,低位在后,所以将byte数组转为int的时候需要倒过来转换:

  
  
  
  
  1.     /**
  2.      * 字节数组和整型的转换,高位在前,适用于读取writeInt的数据
  3.      *
  4.      * @param bytes 字节数组
  5.      * @return 整型
  6.      */
  7.     public static int bytesHighFirstToInt(byte[] bytes, int start)
  8.     {
  9.         int num = bytes[start + 3] & 0xFF;
  10.         num |= ((bytes[start + 2] << 8) & 0xFF00);
  11.         num |= ((bytes[start + 1] << 16) & 0xFF0000);
  12.         num |= ((bytes[start] << 24) & 0xFF000000);
  13.         return num;
  14.     }

你可能感兴趣的:(NIO对字节流的读写)