java - 截断内存映射文件

标签 java linux memory-management io memory-mapped-files

我正在为索引文件使用内存映射 IO,但问题是如果文件大部分是空的,我无法调整文件大小。

之前的某处:

MappedByteBuffer map = raf.getChannel().map(MapMode.READ_WRITE, 0, 1 << 30);
raf.close();
// use map
map.force();
map = null;

调整大小:

for (int c = 0; c < 100; c++) {
    RandomAccessFile raf = new RandomAccessFile(indexFile, "rw");
    try {
        raf.setLength(newLen);
        if (c > 0) LOG.warn("used " + c + " iterations to close mapped byte buffer");
        return;
    } catch (Exception e) {
        System.gc();
        Thread.sleep(10);
        System.runFinalization();
        Thread.sleep(10);
    } finally {
        raf.close();
    }
}

当使用 Windows 或 Linux 32 位时,我经常遇到取消映射问题,但在 64 位 Linux 生产环境中,一切似乎都在没有警告的情况下运行,但文件保持原始大小。

谁能解释为什么会发生这种情况和/或如何解决这个问题?

最佳答案

您的问题是您正在使用不可靠的方法来关闭映射的字节缓冲区(对 System.gc()System.runFinalization() 的一百次调用并不能为您提供任何保证)。不幸的是,在 Java API 中没有可靠的方法可以做到这一点,但在 Sun JVM(也许还有其他一些)上,您可以使用以下代码:

public void unmapMmaped(ByteBuffer buffer) {
  if (buffer instanceof sun.nio.ch.DirectBuffer) {
    sun.misc.Cleaner cleaner = ((sun.nio.ch.DirectBuffer) buffer).cleaner();
    cleaner.clean();
  }
}

当然它依赖于 JVM,如果 Sun 决定更改 sun.nio.ch.DirectBuffer,您应该准备好修复您的代码或 sun.misc.Cleaner以一种不兼容的方式(但实际上我不相信这会发生)。

关于java - 截断内存映射文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5989267/

相关文章:

linux - perf stat 为每次运行提供不同数量的指令

c++ - 这个内存分配分配在哪里?

ios - ARCGIS iOS - map 图层内存问题

java - 不幸的是,项目已在 android studio 中停止

java - Mysql无法将具有特定字符的值存储到数据库

java - 格式化输出,保留三位小数

linux - ">/dev/null 2>&1"和 "2>/dev/null >&2"有什么区别

php - 检查应用程序在哪个服务器上运行?

c++ - 在 32 位 XP 上分配超过 1 GB 的内存

java - Selenium - 从两个显示器捕获屏幕截图