使用java.io.File的renameTo方法移動文件失敗的問題

今天線上發現一個問題,發現一個定時移動文件的業務沒有正常執行,結合日誌和代碼發現,移動文件是使用File類的renameTo方法,可是方法返回的都是false,表示文件移動失敗。
出現這個問題我第一反應是否是文件權限的問題,可是和運維研究後發現的確不是權限致使的。既然不是權限的問題,那就看看renameTo的實現吧,查看源碼發現該方法最終是經過一個本地方法實現的,看不到咋寫的。
網上查了一下renameTo這個方法,發現這個方法確實存在一些問題,就是在不一樣的文件系統中移動是不會成功的。由於測試環境並未出現這個問題,我就把生產環境和測試環境對比了下,發現測試環境下,文件自己的目錄和要移動到的目錄是在/home下,而生產環境中,文件自己目錄是在/home下,要移動到的目錄都是在/data下。因而用df命令查看了一下,發現 /home的文件系統是/dev/sda3,類型是xfs的,/data的文件系統是/dev/sdb1,類型是ext4。java


既然是這樣那就寫個demo在本身的虛擬機上驗證一下是否是這個緣由致使的。apache

1.首先找兩個文件系統不同的目錄,命令df -T
咱們用/tmp 和 /run 做爲測試目錄。
2.測試代碼運維

import java.io.File;
/**
* 文件移動方法測試
*/
public class FileTest {
  public static void main(String[] args) {
     String filePath="/tmp/test.txt";
     File file = new File(filePath);
     boolean b = file.renameTo(new File("/run/test.txt"));
     System.out.println(b);
 }
}

3.編譯運行工具

javac FileTest.java
java FileTest

運行結果輸出false,文件也確實未移動成功測試

解決方法:使用apache的commons-io包中的工具類的進行文件移動。
1.測試代碼:ui

import org.apache.commons.io.FileUtils;
import java.io.File;
import java.io.IOException;
/**
* 文件移動方法測試
*/
public class FileTest {
  public static void main(String[] args) {
     String filePath="/tmp/test.txt";
     File file = new File(filePath);
     boolean b = file.renameTo(new File("/run/test.txt"));
     System.out.println(b);
     //使用apache的FileUtils工具
     try {
         FileUtils.moveFile(file,new File("/run/test.txt"));
         System.out.println("success");
     } catch (IOException e) {
         e.printStackTrace();
     }
 }
}

2.編譯運行日誌

javac -cp /root/jar/commons-io-2.4.jar FileTest.java
java -cp /root/jar/commons-io-2.4.jar: FileTest

運行結果成功移動文件
3.apache的FileUtils移動文件方法的主要實現以下:code

//先使用renameTo方法進行移動
boolean rename = srcFile.renameTo(destFile);
if (!rename) {
	//renameTo移動失敗,就複製文件,而後刪除原文件
	copyFile( srcFile, destFile );
	if (!srcFile.delete()) {
	FileUtils.deleteQuietly(destFile);
	throw new IOException("Failed to delete original file '" + srcFile +
		"' after copy to '" + destFile + "'");
	}
}

總結:blog

  1. 文件移動最好不要使用Java的renameTo方法,而是應該使用apache的commons-io包,固然也能夠本身封裝相似的方法。
  2. renameTo方法移動失敗是文件系統不一樣形成的,補充測試發現不一樣的文件系統,就算類型相同,移動也會失敗。
相關文章
相關標籤/搜索