請慎用java的File#renameTo(File)方法

轉載地址:http://xiaoych.iteye.com/blog/149328linux

 

之前我一直覺得File#renameTo(File)方法與OS下面的 move/mv 命令是相同的,能夠達到更名、移動文件的目的。不事後來常常發現問題:File#renameTo(File)方法會返回失敗(false),文件沒有移動,又查不出緣由,再後來乾脆棄用該方法,本身實現一個copy方法,問題卻是再也沒有出現過。 

昨天老闆同窗又遇到這個問題,File#renameTo(File)方法在windows下面工做的好好的,在linux下偶爾又失靈了。回到家我掃了一遍JDK中File#renameTo(File)方法的源代碼,發現它調用的是一個本地的方法(native method),沒法再跟蹤下去。網上有人說該方法在window下是正常的,在linux下面是不正常的。這個很難說通,SUN不可能搞出這種平臺不一致的代碼出來啊。 

後面在SUN的官方論壇上看到有人提到這個問題「works on windows, don't work on linux」,後面有人回覆說是「file systems」不同。究竟怎麼不同呢?仍是沒有想出來... 

後面在一個論壇裏面發現了某人關於這個問題的闡述: 

apache

引用
In the Unix'esque O/S's you cannot renameTo() across file systems. This behavior is different than the Unix "mv" command. When crossing file systems mv does a copy and delete which is what you'll have to do if this is the case. 

The same thing would happen on Windows if you tried to renameTo a different drive, i.e. C: -> D:



終於明白咯。 

作個實驗: windows


File sourceFile = new File("c:/test.txt");
File targetFile1 = new File("e:/test.txt");
File targetFile2 = new File("d:/test.txt");
System.out.println("source file is exist? " + sourceFile.exists()
+ ", source file => " + sourceFile);
System.out.println(targetFile1 + " is exist? " + targetFile1.exists());
System.out.println("rename to " + targetFile1 + " => "
+ sourceFile.renameTo(targetFile1));
System.out.println("source file is exist? " + sourceFile.exists()
+ ", source file => " + sourceFile);
System.out.println(targetFile2 + " is exist? " + targetFile2.exists());
System.out.println("rename to " + targetFile2 + " => "
+ sourceFile.renameTo(targetFile2));app

 

source file is exist? true, source file => c:\test.txt
e:\test.txt is exist? false
rename to e:\test.txt => false
source file is exist? true, source file => c:\test.txt
d:\test.txt is exist? false
rename to d:\test.txt => truethis

 

注意看結果,從C盤到E盤失敗了,從C盤到D盤成功了。由於個人電腦C、D兩個盤是NTFS格式的,而E盤是FAT32格式的。因此從C到E就是上面文章所說的"file systems"不同。從C到D因爲同是NTFS分區,因此不存在這個問題,固然就成功了。 

果真是不能把File#renameTo(File)看成move方法使用。 

能夠考慮使用apache組織的commons-io包裏面的FileUtils#copyFile(File,File)和FileUtils#copyFileToDirectory(File,File)方法實現copy的效果。至於刪除嘛,我想若是要求不是那麼精確,能夠調用File#deleteOnExit()方法,在虛擬機終止的時候,刪除掉這個目錄或文件。 

BTW:File是文件和目錄路徑名的抽象表示形式,因此有多是目錄,千萬當心。blog

相關文章
相關標籤/搜索