`
xiaoych
  • 浏览: 145945 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

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

    博客分类:
  • java
阅读更多
以前我一直以为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”不一样。究竟怎么不一样呢?还是没有想出来...

后面在一个论坛里面发现了某人关于这个问题的阐述:

引用
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:


终于明白咯。

做个实验:


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));


结果:


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 => true



注意看结果,从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是文件和目录路径名的抽象表示形式,所以有可能是目录,千万小心。
分享到:
评论
16 楼 何飞同学要加油 2014-12-18  
文件系统不一样好像是说 windows和Linux的文件默认为NTFS的 而Linux是ext32的
15 楼 lvxiaoxi 2013-05-14  
还有一种情况,看我的博文:
http://lvxiaoxi.iteye.com/blog/1868440
14 楼 di1984HIT 2013-02-05  
嗯,说的很对啊。
13 楼 xxm7801877 2013-01-05  
                   
12 楼 clican 2008-07-28  
我还碰到过在大压力情况下在windows renameTo有一定概率失败的情况。
在linux操作系统上在不同盘符之间renameTo也会失败。典型的应用场景就是从本地硬盘renameTo到mount的硬盘上或文件系统上。
11 楼 xiaobai1023 2008-07-18  
有时候要注意对文件的引用是否已经关闭了
10 楼 cmoaciopm 2008-01-22  
jdk版本:1.5.0_09
操作系统:WindowsXP sp2
环境:C盘为NTFS分区,F盘为FAT32分区
现象:1.将C盘的文件renameTo到F盘,返回成功且文件移动成功
      2.将F盘的文件renameTo到C盘,返回成功且文件移动成功

并没有出现楼主所说的现象,不知道是不是jdk版本的问题
9 楼 温柔一刀 2008-01-21  
确实是这样的,曾经遇到过。copy文件尽量使用FileUtilsc.opyFileToDirectory(File,File)

还有个问题是在使用webwork的fileUpload拦截器的时候尽量不要使用execAndWait拦截器,两个拦截器会开启两个线程,很可能会删掉还没有处理的临时上传文件文件而导致文件找不到。
8 楼 zhanjianhua 2008-01-10  
File oldFile = new File(souceFile);
File newFile = new File(souceFile.replace(".", "del."));
oldFile.renameTo(newFile);
这样修改文件名,oldFile, newFile怎么关闭,没看到JDK中有这方面的方法?
还有这种修改文件名,无限制的修改下去,并且修改不同的文件名,会报java.io.IOException: java.io.IOException: Too many open files的错不
7 楼 Jonney 2007-12-22  
很久以前看过一本书介绍了java里面的很多个陷阱,
从此我一直把java.io.File当成是java.io.Filename来看待。
6 楼 linxizeng 2007-12-21  
就是喜欢这种实际经验的贴...
5 楼 xiaoych 2007-12-21  
zhangzhaofeng 写道
对楼主说的进行了在Windows下面的测试
我自己发现有3种情况会在
不同盘之间的copy中出错

第一:源文件不存在
第二:目标文件存在
第三:都不存在

楼主的那种rename to d:\test1.txt => false
我怀疑是第一种情况
应该和盘符没有关系


我改正了一下逻辑错误,多谢你的提醒,呵呵
4 楼 xiaoych 2007-12-21  
是的,File#renameTo(File)方法 有很多种情况会返回false或者抛出异常。我做的那个测试只是想表达其中的一种情况:

就是源文件存在,目标文件不存在,对目标目录有写入权限(由于是windows下面的administrator,所以默认会有),但是"file systems"不同,也会返回失败。

这是我们最需要注意的,也就是我写这个的原因。
3 楼 zhangzhaofeng 2007-12-21  
可能我的判断是有问题的
rename好奇怪
2 楼 zhangzhaofeng 2007-12-21  
对楼主说的进行了在Windows下面的测试
我自己发现有3种情况会在
不同盘之间的copy中出错

第一:源文件不存在
第二:目标文件存在
第三:都不存在

楼主的那种rename to d:\test1.txt => false
我怀疑是第一种情况
应该和盘符没有关系
1 楼 myreligion 2007-12-19  
如果目标文件已经存在,rename一般也会失败,返回false。这个东西还是很靠不住的,不知道实现者是怎么想的。

相关推荐

    java file类的方法

    ### Java File 类的方法详解 #### 一、简介 在Java编程语言中,`java.io.File`类是一个非常重要的类,它提供了对文件和目录路径名的抽象表示,并且支持一些基本的操作,例如创建、删除文件或目录等。本文将详细...

    java中File类的使用方法 File类的

    * `public boolean renameTo(File f)`: 将文件重命名为指定的文件。 * `public File[] listRoots()`: 获取机器的盘符。 * `public String[] list()`: 列出文件夹下的文件和文件夹。 * `public String[] list...

    java中File类总结

    18. **`renameTo(File dest)`**:重命名文件或移动文件到另一个位置。 19. **`setReadOnly()`**:设置文件为只读。 20. **`toString()`**:返回描述文件状态的字符串。 21. **`toURL()`**:将文件转换为URL对象。 #...

    Better File Rename 绿色汉化版

    Better File Rename 5.7 绿色汉化版 Better File Rename 是一款强大易用的文件批量改名软件。内置最全面的重命名选项。 现已支持64位操作系统。 5.7 英文原版更新说明: ① 添加了 PDF 格式的用户手册 ② 在所有...

    java file 文件操作例子

    boolean renamed = file.renameTo(newFile); if (renamed) { System.out.println("文件重命名成功"); } else { System.out.println("文件重命名失败"); } ``` 6. **路径操作**:`File`类提供了一些方法来处理路径...

    java File类文件的使用

    - `renameTo(File dest)`:将文件或目录重命名为指定的新名称。 3. **文件属性获取**: - `exists()`:判断文件或目录是否存在。 - `isFile()` 和 `isDirectory()`:分别判断是否为文件和目录。 - `canRead()` ...

    Better File Rename v5.7

    "Better File Rename v5.7" 是一个专为Windows用户设计的应用软件,它极大地提升了文件重命名的效率和便利性。作为一个资源管理器的扩展外壳,该工具无缝集成到Windows操作系统中,使得用户在不离开熟悉的文件管理...

    File_java_

    - `renameTo(File dest)`:将文件或目录重命名为指定的新名称。此操作可能因操作系统而异,不总是跨文件系统支持。 - `delete()`:删除文件或空目录。如果文件是目录并且非空,该方法将失败。 4. **属性查询**: ...

    Java 语言File类的详解

    - `boolean renameTo(File dest)`:将文件或目录重命名为指定的新名称,如果目标文件已存在,重命名可能失败。 **六、文件的遍历** - `File[] listFiles()`:列出当前目录下的所有文件和目录,返回File数组。 - `...

    Day36 Java的file类

    - **`boolean renameTo(File dest)`**:重命名文件。 4. **文件判断** - **`boolean isDirectory()`**:判断是否为目录。 - **`boolean isFile()`**:判断是否为文件。 #### 四、具体案例分析 ##### 需求1:...

    linux下用renameTo方法修改java web项目中文件夹名称的实例

    Linux下用renameTo方法修改Java Web项目中文件夹名称的实例 在 Linux 环境中,修改 Java Web 项目中的文件夹名称是一个常见的问题。今天,我们将分享一个使用 renameTo 方法修改 Java Web 项目中文件夹名称的实例。...

    Quick File Rename 1.2

    Quick File Rename 1.2是一款轻巧而高效的文件批量重命名工具,专为需要快速更改大量文件后缀名的用户设计。它以其简洁的操作界面和强大的功能,为用户提供了便捷的文件管理体验,尤其对比一些体积庞大、功能复杂的...

    Better File Rename文件重命名工具

    "Better File Rename"是一款高效的文件重命名工具,专为用户提供了丰富多样的文件批量重命名功能,极大地提升了在日常工作中处理大量文件的效率。这款工具不仅适用于个人用户,也广泛应用于企业和团队,使得文件管理...

    java-io-file类笔记

    这些方法是`File`类的基础操作,但请注意,它们都是异步的,不保证立即反映文件系统的变化。同时,某些操作可能会受操作系统权限和安全策略的影响。在实际应用中,需要进行适当的错误处理和异常捕获。

    java中IO流里面的关于File的讲解源码

    例如,`exists()`检查文件是否存在,`createNewFile()`尝试创建新文件,`delete()`删除文件,`renameTo()`重命名文件。同时,`length()`返回文件大小,`lastModified()`获取文件最后修改时间。 当涉及到IO流时,...

    rename file

    【标题】:“rename file”指的是批量重命名文件的程序,这是一个在IT领域常见的操作,尤其在数据管理和自动化脚本编写中。批量重命名文件能够帮助用户高效地整理大量文件,使其具有更统一、更具描述性的名称,从而...

    java修改文件后缀

    这可以通过`renameTo()`方法实现,但要注意,`renameTo()`在某些情况下可能不跨文件系统工作,例如从网络驱动器到本地驱动器: ```java boolean isRenamed = oldFile.renameTo(newFile); if (isRenamed) { System....

Global site tag (gtag.js) - Google Analytics