Windows上可靠的File.renameTo()替代方法?

File.renameTo()似乎Java 是有问题的,尤其是在Windows上。如API文档所述,


此方法的行为的许多方面本质上与平台有关:重命名操作可能无法将文件从一个文件系统移动到另一个文件系统,它可能不是原子的,并且如果具有目标抽象路径名的文件可能无法成功已经存在。应该始终检查返回值,以确保重命名操作成功。


就我而言,作为升级过程的一部分,我需要移动(重命名)可能包含千兆字节数据(很多子目录和大小不同的文件)的目录。移动始终在同一分区/驱动器中完成,因此并不需要物理上移动磁盘上的所有文件。


这里不应该有任何文件锁定的目录中的内容被移动了,不过,很多时候,renameTo()无法完成其工作,并返回false。(我只是猜测也许某些文件锁在Windows上会任意地到期。)


当前,我有一个使用复制和删除的后备方法,但是这种方法很麻烦,因为它可能要花费很多时间,具体取决于文件夹的大小。我也在考虑简单地记录一个事实,即用户可以手动移动文件夹,以免潜在地等待数小时。但是正确的方法显然是自动而快速的。


所以我的问题是,您是否知道另一种可靠的方法,可以在Windows上使用Java(使用普通的JDK或某些外部库)进行快速移动/重命名。或者,如果您知道一种检测和释放给定文件夹及其所有内容(可能是成千上万个单独文件)的任何文件锁的简便方法,那也可以。



慕田峪4524236
浏览 916回答 3
3回答

沧海一幻觉

另请参见Files.move()JDK 7中的方法。一个例子:String fileName = "MyFile.txt";try {    Files.move(new File(fileName).toPath(), new File(fileName).toPath(), java.nio.file.StandardCopyOption.REPLACE_EXISTING);} catch (IOException ex) {    Logger.getLogger(SomeClass.class.getName()).log(Level.SEVERE, null, ex);}

泛舟湖上清波郎朗

对于它的价值,还有一些其他概念:在Windows上,renameTo()即使目标目录存在,即使它为空,似乎也会失败。就像我在Linux上尝试过的renameTo()那样,只要目标为空,只要目标存在,它就会成功,这使我感到惊讶。(显然,我不应该假设这种事情在所有平台上都可以起作用;这正是Javadoc警告的。)如果您怀疑可能存在一些挥之不去的文件锁,请稍等片刻再进行移动/重命名可能会有所帮助。(在安装程序/升级程序的某一点上,我们添加了一个“睡眠”操作和一个不确定的进度条,持续了大约10秒钟,因为某些文件可能挂着服务)。甚至可以执行一个简单的重试机制try renameTo(),然后等待一段时间(可能逐渐增加),直到操作成功或达到某个超时为止。就我而言,大多数问题似乎都可以通过兼顾以上两者来解决,因此我们根本不需要进行本机内核调用或诸如此类的事情。

LEATH

就我而言,这似乎是我自己的应用程序中的死对象,该对象保留了该文件的句柄。因此,该解决方案对我有用:for (int i = 0; i < 20; i++) {&nbsp; &nbsp; if (sourceFile.renameTo(backupFile))&nbsp; &nbsp; &nbsp; &nbsp; break;&nbsp; &nbsp; System.gc();&nbsp; &nbsp; Thread.yield();}优点:速度非常快,因为没有Thread.sleep()带有特定的硬编码时间。缺点:20的限制是一些硬编码的数字。在我所有的测试中,i = 1就足够了。但可以肯定的是,我将其保留为20岁。
打开App,查看更多内容
随时随地看视频慕课网APP