美文网首页Java学习笔记
File练习--实现目录(文件)的复制,剪贴,删除

File练习--实现目录(文件)的复制,剪贴,删除

作者: 比轩 | 来源:发表于2016-11-09 20:27 被阅读73次

    一个小练习,使用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);
                    }
                }
            }
        }
        
    }
    
    
    

    相关文章

      网友评论

        本文标题:File练习--实现目录(文件)的复制,剪贴,删除

        本文链接:https://www.haomeiwen.com/subject/aartpttx.html