一个小练习,使用java实现目录(包括文件和子目录或者子文件在内)的复制,剪贴和删除,基本实现思路就是利用递归,由内往外逐级删除。
其实本来是没有想到写这个练习的,谁知道今天写了一个File的demo,产生了bug,导致在一个目录下面新建目录,新建的目录下再新建目录,导致一直到栈内存溢出才停止了。然后手动去尝试删除那些目录的时候问题来了!Windows提示,文件路径长度超过NTFS支持的最大长度,让我手动把最内层的文件夹复制出来,变短一点再删除。尼玛,不停双击再双击,不知道多少层了都找翻不到底层。所以就放弃了,直接写了个demo处理了下这个问题,顺便做了一下练习。
具体代码里面有详细的注释和一些解释,直接复制javac就能跑,删除和剪贴功能清慎用,代码删除是没有办法撤销的,误删要找回来是比较麻烦的。
package file;
/*
* 递归练习实现目录的删除,复制,剪贴
*/
import java.io.*;
public class Digui {
public static void main(String[] args) {
//deleteDir(new File("E:\\demo\\a\\a"));
//cut(new File("E:\\demo"), new File("E:\\demo1"));
//copyDir(new File("E:\\demo\\b\\"),new File("E:\\demo1"));
}
/*
* 调用删除和复制,实现剪贴
*/
public static void cut(File dirFrom, File dirTo) {
// 先复制完成
copy(dirFrom, dirTo);
// 删除原目录
delete(dirFrom);
}
/*
* 递归实现目录的删除(包括目录下的子目录和文件)
*/
public static void delete(File dir) {
// 如果传入的对象为空,直接结束方法
if (dir == null)
return;
// 如果传入的目录不存在,直接结束方法
if (!dir.exists())
return;
// 如果是文件,直接删除
if (dir.isFile()) {
dir.delete();
return;
} else { // 如果是目录,判断是不是空目录,空目录直接删除,不是空目录进去递归调用
File[] dirs = dir.listFiles();
if (dirs == null) {
dir.delete();
return;
} else {
for (File f : dirs) {
delete(f);
}
}
//最后再删除父目录本身,因为此时父目录为空,才可以删除
dir.delete();
}
}
/* 复制目录
*
* 目标:把原目录(包括源目录在内)下的子目录和文件全部复制到新目标目录下
*
* 思路(递归实现目录的复制 ):
* 1.方法的前面先 进行健壮性处理,空对象等等
* 2.然后分情况判断,原目录是一个文件,直接复制完成结束方法,是目录进行以下步骤
* 3.获取源目录,创建新目录
* 4.逐个创建子目录,如果是文件,创建完成后,把数据copy过来
* 5.如果是子目录是路径,则递归调用方法,传入新的源目录和目标目录file对象,从3开始继续
*
* 健壮性思路:
* 源目录和目标目录为空对象,直接报runtime异常
* 源目录和目标目录为同一个对象,直接报runtime异常
* 目标目录不存在,尝试创建,创建失败报runtime异常,若成功提示新建了目录后继续执行
* 目标目录是文件时,复制在此文件所在的目录下
*/
public static void copy(File dirFrom, File dirTo) {
// 健壮性处理
// 空对象,直接报错
if (dirFrom == null || dirTo == null)
throw new RuntimeException("文件对象为空!");
// 目标目录的原目录相同,直接返回
if (dirFrom == dirTo)
throw new RuntimeException("不能原地复制");
// 目标目录不存在处理
if (!dirTo.exists()) {
if (dirTo.mkdirs()) {
System.out.println("目标目录不存在,已创建");
} else {
throw new RuntimeException("目标目录不存在,尝试创建失败");
}
}
// 目标目录是一个文件,复制在此文件的目录下
if(dirTo.isFile())
dirTo = dirTo.getParentFile();
// 原目录是一个文件,直接把文件复制过去即可
if (dirFrom.isFile()) {
BufferedInputStream bis1 = null;
BufferedOutputStream bos1 = null;
File newFile1 = new File(dirTo, dirFrom.getName());
try {
newFile1.createNewFile();
bis1 = new BufferedInputStream(new FileInputStream(dirFrom));
bos1 = new BufferedOutputStream(new FileOutputStream(newFile1));
int len1 = 0;
byte[] data1 = new byte[1024];
while ((len1 = bis1.read(data1)) != -1) {
bos1.write(data1, 0, len1);
bos1.flush();
}
} catch (IOException e) { //异常处理,报异常直接结束程序
throw new RuntimeException("文件复制失败");
} finally { //关闭流对象
try {
bos1.close();
} catch (IOException e) {
throw new RuntimeException("文件对象关闭失败");
}
try {
bis1.close();
} catch (IOException e) {
throw new RuntimeException("文件对象关闭失败");
}
}
return; // 因为原file是一个文件,所以复制完成后,直接结束方法
} else {//原目录不是目录,进入下面的代码
//在目标目录下创建和源目录name相同的file对象,并实例化
File newCopyDir = new File(dirTo, dirFrom.getName());
newCopyDir.mkdir();
//获取源目录下的子目录
File[] fileFrom = dirFrom.listFiles();
//如果源目录是一个空的目录,则直接结束掉方法即可
//因为上面已经创建了新的目录的,也就是说复制已经完成
if(fileFrom.length == 0)
return;
for (File f : fileFrom) {
if (f.isFile()) {
// 判断子目录下的File如果是一个文件,则在新目录下也创建一个相同的文件
File newFile = new File(newCopyDir, f.getName());
BufferedInputStream bis = null;
BufferedOutputStream bos = null;
try {
newFile.createNewFile();
bis = new BufferedInputStream(new FileInputStream(f));
bos = new BufferedOutputStream(new FileOutputStream(newFile));
int len = 0;
byte[] data = new byte[1024];
while ((len = bis.read(data)) != -1) {
bos.write(data, 0, len);
bos.flush();
}
} catch (IOException e) {
throw new RuntimeException("文件复制失败");
} finally { //在finally下关闭两个流对象,并做异常处理
try {
bos.close();
} catch (IOException e1) {
throw new RuntimeException("文件对象关闭失败");
}
try {
bis.close();
} catch (IOException e1) {
throw new RuntimeException("文件对象关闭失败");
}
}
} else { // 子目录不是一个文件,就采取递归模式,继续进行
copy(f, newCopyDir);
}
}
}
}
}