使用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方法移动失败是文件系统不一样形成的,补充测试发现不一样的文件系统,就算类型相同,移动也会失败。
相关文章
相关标签/搜索