MappedByteBuffer使用注意点

标签:

it

 

最近使用MD5进行大文件验证,固使用NIO这种高效率的模式来进行文件映射:

Java代码
  1. FileInputStream in = newFileInputStream(file);   
  2. FileChannel ch = in.getChannel();   
  3. MappedByteBuffer byteBuffer = ch.map(FileChannel.MapMode.READ_ONLY,0, file.length());   
  4. messageDigest.update(byteBuffer);   
  5. String md5 = bufferToHex(messageDigest.digest());   
  6. ch.close();   
  7. in.close();  

本来想如果文件md5与数据库存储的值不同就删掉该文件的,结果出现了文件无法删除的情况。

抛出的违例
java.io.FileNotFoundException: E:\hello.jar (请求的操作无法在使用用户映射区域打开的文件上执行。)

后来经查,原来是当使用 FileChannel.map 方法时,MappedByteBuffer 已经在系统内占用了一个句柄,而使用 FileChannel.close 方法是无法释放这个句柄的,且FileChannel有没有提供类似 unmap 的方法,因此会出现无法删除文件的情况。

此问题一直让我很郁闷,后来到网上查询到两种方法...

第一种:

Java代码
  1. AccessController.doPrivileged(newPrivilegedAction() {   
  2.   publicObject run() {   
  3.     try{   
  4.        Method getCleanerMethod = buffer.getClass().getMethod("cleaner",newClass[0]);   
  5.        getCleanerMethod.setAccessible(true);   
  6.        sun.misc.Cleaner cleaner = (sun.misc.Cleaner)   
  7.        getCleanerMethod.invoke(byteBuffer,newObject[0]);   
  8.        cleaner.clean();   
  9.      } catch(Exception e) {   
  10.        e.printStackTrace();   
  11.      }   
  12.     returnnull;   
  13.    }   
  14. });  

 

publicstaticvoid clean(ByteBuffer bb){ 
   
if(bb== null)return; 
   
Cleaner cleaner= ((DirectBuffer) bb).cleaner(); 
   
if(cleaner !=null) cleaner.clean(); 
} 

此种方法需要JDK支持,我用的是JRE 1.6,提示没有 sun.misc.Cleaner 类,JDK包太大,项目又不让用。

第二种方法是显性设置byteBuffer为null,并调用GC,没什么实际意义。

实在没招了,又回来使用InputStream了...

Java代码
  1. FileInputStream in = newFileInputStream(file);   
  2. ByteArrayOutputStream out = newByteArrayOutputStream((int)file.length());   
  3. byte[] cache =newbyte[1048576];   
  4. for(inti = in.read(cache);i != -1;i = in.read(cache)){   
  5.    out.write(cache, 0, i);   
  6. }   
  7. in.close();   
  8. out.close();   
  9. messageDigest.update(out.toByteArray());   
  10. md5 = bufferToHex(messageDigest.digest());  
 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用MappedByteBuffer可以将文件直接映射到内存中,通过内存操作来读文件,从而提高性能。下面是使用MappedByteBuffer的示例代码: ```java try { // 创建 RandomAccessFile 对象 RandomAccessFile file = new RandomAccessFile("path/to/file.txt", "rw"); // 获取文件通道 FileChannel channel = file.getChannel(); // 将文件映射到内存中 MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_WRITE, 0, channel.size()); // 读取数据 byte[] data = new byte[buffer.limit()]; buffer.get(data); // 入数据 String content = "Hello, World!"; buffer.put(content.getBytes()); // 刷新缓冲区内容到磁盘 buffer.force(); // 关闭资源 channel.close(); file.close(); } catch (IOException e) { e.printStackTrace(); } ``` 在上述代码中,首先通过RandomAccessFile对象获取文件通道,然后通过`map()`方法将文件映射到内存中的MappedByteBuffer对象。通过该对象可以直接对文件进行读操作。 读取数据时,可以通过`get()`方法从MappedByteBuffer中获取字节数据。入数据时,可以通过`put()`方法将字节数据MappedByteBuffer。 需要注意的是,在进行入操作后,最好调用`force()`方法刷新缓冲区内容到磁盘,以确保数据被持久化保存。 最后,记得关闭资源,释放系统资源。 使用MappedByteBuffer可以减少磁盘I/O次数,提高读性能,但需要注意内存映射文件的大小限制,过大的文件可能会导致内存溢出。此外,MappedByteBuffer适用于较大的文件,对于小文件可能带来较小的性能提升。因此,应根据具体情况评估是否使用MappedByteBuffer来优化RandomAccessFile的性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值