检查文件是否完整写入

目前我正在开发一个项目,它在其中一个例程中处理源目录中的文件。 有一个Java进程正在查找指定的目录,并尝试读取和处理文件(如果存在)。 文件退出大并由其他第三方进程更新。 问题是如何检查文件是否完整写入? 我正在尝试使用file.length()但看起来即使写入过程尚未完成它也会返回实际大小。 我觉得解决方案应该依赖于平台。 任何帮助,将不胜感激。

更新:这个问题与副本没有什么不同,但它有一个高评级的工作代码片段的答案。

生成器进程在完成写入后是否关闭文件? 如果是这样,如果生产者进程仍在生成,则尝试使用独占锁打开使用者进程中的文件将失败。

我得到了解决方案:

 private boolean isCompletelyWritten(File file) { RandomAccessFile stream = null; try { stream = new RandomAccessFile(file, "rw"); return true; } catch (Exception e) { log.info("Skipping file " + file.getName() + " for this iteration due it's not completely written"); } finally { if (stream != null) { try { stream.close(); } catch (IOException e) { log.error("Exception during closing file " + file.getName()); } } } return false; } 

感谢@cklab和@Will以及其他建议以“独家锁定”方式查看的人。 我刚刚在这里发布了代码,让其他人对使用它感兴趣。 我相信@tigran建议的重命名解决方案也有效,但纯Java解决方案对我来说更合适。

PS最初我使用FileOutputStream而不是RandomAccessFile但它锁定正在写入的文件。

我在过去使用Windows的这种情况下使用的一个简单解决方案是使用boolean File.renameTo(File)并尝试将原始文件移动到单独的暂存文件夹:

 boolean success = potentiallyIncompleteFile.renameTo(stagingAreaFile); 

如果successfalse ,则仍然会写入potentiallyIncompleteFile

我不认为有一个通用的解决方案。 查找文件大小是错误的,因为某些应用程序可以在任何写入调用之前设置文件大小。 其中一种可能性是使用锁定。 这将要求编写器产生写锁(或独占锁)。 如果你不能修改编写器,那么你可以使用操作系统提供的工具,比如Linux上的fuser,看看有没有一个进程仍然可以访问该文件。

如果您计划在单个平台上使用此代码,则可以使用NIO的FileLock工具 。 但请仔细阅读文档,并注意在许多平台上,锁只是建议性的。

另一种方法是让一个进程使用您的进程无法识别的名称写入文件,然后在写入完成时将文件重命名为可识别的名称。 在大多数平台上,如果源和目标是相同的文件系统卷,则重命名操作是primefaces操作。

如果您可以使用Java 1.7,请查看NIO工具,特别是java.nio.channels.FileChannel

这是一个锁定文件并读取它的示例。