java - 使流无效而不关闭

标签 java file-io file-descriptor fileinputstream finalizer

这是 anonymous file streams reusing descriptors 的后续内容

根据我之前的问题,我不能依赖这样的代码(目前恰好在 JDK8 中工作):

RandomAccessFile  r = new RandomAccessFile(...);

FileInputStream f_1 = new FileInputStream(r.getFD());
                      // some io, not shown
                f_1 = null;

                f_2 = new FileInputStream(r.getFD());
                      // some io, not shown
                f_2 = null;

                f_3 = new FileInputStream(r.getFD());
                    // some io, not shown
                f_3 = null;

但是,为了防止意外错误并作为 self 文档的一种形式,我想在使用完每个文件流后使其无效 - 而不关闭底层文件描述符。

每个 FileInputStream 都是独立的,其位置由 RandomAccessFile 控制。我共享相同的 FileDescriptor,以防止多次打开同一路径而产生任何竞争条件。当我使用完一个 FileInputStream 后,我想使其无效,以便在使用第二个 FileInputStream 时不会意外读取它(这将导致第二个 FileInputStream 跳过数据)。

我该怎么做?

注释:

  • 我使用的库需要与 java.io.* 兼容
  • 如果您建议一个库(如果可能的话,我更喜欢内置 Java 语义),它必须对于 Linux(主要目标)通用(打包)并且可以在 Windows(实验目标)上使用
  • 但是,Windows 支持并不是绝对必需的

编辑:为了回复评论,这是我的工作流程:

RandomAccessFile r = new RandomAccessFile(String path, "r");

int header_read;
int header_remaining = 4; // header length, initially

byte[]     ba = new byte[header_remaining];
ByteBuffer bb = new ByteBuffer.allocate(header_remaining);

while ((header_read = r.read(ba, 0, header_remaining) > 0) {
    header_remaining -= header_read;
    bb.put(ba, 0, header_read);
}

byte[] header = bb.array();

// process header, not shown
// the RandomAccessFile above reads only a small amount, so buffering isn't required

r.seek(0);

FileInputStream f_1 = new FileInputStream(r.getFD());

Library1Result result1 = library1.Main.entry_point(f_1)

// process result1, not shown
// Library1 reads the InputStream in large chunks, so buffering isn't required
// invalidate f_1 (this question)

r.seek(0)

int read;
while ((read = r.read(byte[4096] buffer)) > 0 && library1.continue()) {
    library2.process(buffer, read);
}

// the RandomAccessFile above is read in large chunks, so buffering isn't required
// in a previous edit the RandomAccessFile was used to create a FileInputStream. Obviously that's not required, so ignore

r.seek(0)

Reader r_1 = new BufferedReader(new InputStreamReader(new FileInputStream(r.getFD())));

Library3Result result3 = library3.Main.entry_point(r_2)

// process result3, not shown
// I'm not sure how Library3 uses the reader, so I'm providing buffering
// invalidate r_1 (this question) - bonus: frees the buffer

r.seek(0);

FileInputStream f_2 = new FileInputStream(r.getFD());

Library1Result result1 = library1.Main.entry_point(f_2)

// process result1 (reassigned), not shown
// Yes, I actually have to call 'library1.Main.entry_point' *again* - same comments apply as from before
// invalidate f_2 (this question)
//
// I've been told to be careful when opening multiple streams from the same
// descriptor if one is buffered. This is very vague. I assume because I only
// ever use any stream once and exclusively, this code is safe.
//

最佳答案

纯 Java 解决方案可能是创建一个转发装饰器,用于检查每个方法调用是否验证了流。对于 InputStream 这个装饰器可能如下所示:

public final class CheckedInputStream extends InputStream {
  final InputStream delegate;
  boolean validated;

  public CheckedInputStream(InputStream stream) throws FileNotFoundException {
    delegate = stream;
    validated = true;
  }

  public void invalidate() {
    validated = false;
  }

  void checkValidated() {
    if (!validated) {
      throw new IllegalStateException("Stream is invalidated.");
    }
  }

  @Override
  public int read() throws IOException {
    checkValidated();
    return delegate.read();
  }

  @Override
  public int read(byte b[]) throws IOException {
    checkValidated();
    return read(b, 0, b.length);
  }

  @Override
  public int read(byte b[], int off, int len) throws IOException {
    checkValidated();
    return delegate.read(b, off, len);
  }

  @Override
  public long skip(long n) throws IOException {
    checkValidated();
    return delegate.skip(n);
  }

  @Override
  public int available() throws IOException {
    checkValidated();
    return delegate.available();
  }

  @Override
  public void close() throws IOException {
    checkValidated();
    delegate.close();
  }

  @Override
  public synchronized void mark(int readlimit) {
    checkValidated();
    delegate.mark(readlimit);
  }

  @Override
  public synchronized void reset() throws IOException {
    checkValidated();
    delegate.reset();
  }

  @Override
  public boolean markSupported() {
    checkValidated();
    return delegate.markSupported();
  }
}

您可以像这样使用它:

CheckedInputStream f_1 = new CheckedInputStream(new FileInputStream(r.getFD()));
                      // some io, not shown
                   f_1.invalidate();

                   f_1.read(); // throws IllegalStateException

关于java - 使流无效而不关闭,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35137475/

相关文章:

java - 为什么 CompletableFuture allOf 方法会进行二分查找?

java - 我需要同步这个吗

c++ - 读取文本文件 : Read multiple values if present

java - 创建/写入/读取 Jar 存档外部的文本文件

c - 将文件读入缓冲区,但调用 fseek 后程序在 fread 处崩溃

linux - Linux 中的 select() 和 poll()

java - Play 请求中的模拟查询字符串

java - 使用 FirebaseUI 在 Android TextView 中绑定(bind)嵌套的 JSON

unix - 我如何从 Rust 中的特定原始文件描述符中读取?

c - open() 和 fopen() 的应用