今天線上發現一個問題,發現一個定時移動文件的業務沒有正常執行,結合日誌和代碼發現,移動文件是使用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