Java利用文件输入输出流实现文件夹内所有文件拷贝到另一个文件夹

 更新时间:2018年03月15日 10:07:33   作者:yongh701  
这篇文章主要介绍了Java实现文件夹内所有文件拷贝到另一个文件夹,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

一、基本目标

使用Java完成如下的操作:

把一个文件夹内的所有文件拷贝的另一的文件夹,例如,在F盘中有a与b两个文件夹:

f:/a里面有一堆文件,运行Java程序之后就会全部复制到f:/b,并且完成重命名,在所有文件之前加rename_的前缀,如果里面有文件夹,则文件夹不重命名,里面的文件进行重命名,同样在所有文件之前加rename_的前缀:

二、制作过程

1、首先主函数非常简单,就是调用了上面FileTest类中的copyFolder函数

public class FileCopy { 
 public static void main(String args[]) { 
  new FileTest().copyFolder("f:/a", "f:/b"); 
 
 } 
} 

值得注意的是,这个的传递过去的参数的路径写法,在java中,f:/a是没有问题的,f:\a也是没有问题的,但是由于\在字符串表达的时候,必须转移,所以你必须写成f:\\a

2、整个程序的关键在这个FileTest类中的copyFolder函数,这个类里面就这个函数-_-!而且注意在程序开头引入java.io.*;由于用到了输入输出流

class FileTest { 
 public void copyFolder(String oldPath, String newPath) { 
  try { 
   // 如果文件夹不存在,则建立新文件夹 
   (new File(newPath)).mkdirs(); 
   //读取整个文件夹的内容到file字符串数组,下面设置一个游标i,不停地向下移开始读这个数组 
   File filelist = new File(oldPath); 
   String[] file = filelist.list(); 
   //要注意,这个temp仅仅是一个临时文件指针 
   //整个程序并没有创建临时文件 
   File temp = null; 
   for (int i = 0; i < file.length; i++) { 
    //如果oldPath以路径分隔符/或者\结尾,那么则oldPath/文件名就可以了 
    //否则要自己oldPath后面补个路径分隔符再加文件名 
    //谁知道你传递过来的参数是f:/a还是f:/a/啊? 
    if (oldPath.endsWith(File.separator)) { 
     temp = new File(oldPath + file[i]); 
    } else { 
     temp = new File(oldPath + File.separator + file[i]); 
    } 
     
    //如果游标遇到文件 
    if (temp.isFile()) { 
     FileInputStream input = new FileInputStream(temp); 
     FileOutputStream output = new FileOutputStream(newPath 
       + "/" + "rename_" + (temp.getName()).toString()); 
     byte[] bufferarray = new byte[1024 * 64]; 
     int prereadlength; 
     while ((prereadlength = input.read(bufferarray)) != -1) { 
      output.write(bufferarray, 0, prereadlength); 
     } 
     output.flush(); 
     output.close(); 
     input.close(); 
    } 
    //如果游标遇到文件夹 
    if (temp.isDirectory()) { 
     copyFolder(oldPath + "/" + file[i], newPath + "/" + file[i]); 
    } 
   } 
  } catch (Exception e) { 
   System.out.println("复制整个文件夹内容操作出错"); 
  } 
 } 
} 

可能游标遇到文件部分有点难以理解,其实是这样的,首先设置一个文件的输入流,指定从游标遇到的文件中输入,再指定输出到newPath/rename_旧文件的文件名这个文件目录,之后,设置一个缓冲数组,文件输入流对于自己要读取的文件,每次调用read方法,它都会向后继续上一次读取的位置继续读取缓冲数组bufferarray的长度的内容,把读取到的内容存储到缓冲数组,覆盖缓冲数组之前的所有内容,然后文件输出流会把缓冲数组的所有内容输出的指定的位置,直到文件输入流遇到了-1。

至于文件输入流为何能这样按顺序,每次都会向后继续上一次读取的位置继续读取,那是因为当要进行文件的读取,Java封装的FileInputStream.read方法也会调用操作系统的API依次读取这些数据。在读取文件数据的时候必须是顺序的,不可能说先读取第一个字节,后读取倒数第二个字节。循环读取的时候就read方法将读取的位置++,因此造成每次read都是顺序读取后面的字节,直到遇到文件末尾标记。

当游标遇到文件夹则重新调用自己完成同样的操作即可,这就是所谓的迭代。

3、因此整个程序如下:

import java.io.*; 
 
/** 
 * 
 * @param oldPath 被拷贝的目录 
 * @param newPath 要拷贝到的目录 
 * 
 */ 
class FileTest { 
 public void copyFolder(String oldPath, String newPath) { 
  try { 
   // 如果文件夹不存在,则建立新文件夹 
   (new File(newPath)).mkdirs(); 
   //读取整个文件夹的内容到file字符串数组,下面设置一个游标i,不停地向下移开始读这个数组 
   File filelist = new File(oldPath); 
   String[] file = filelist.list(); 
   //要注意,这个temp仅仅是一个临时文件指针 
   //整个程序并没有创建临时文件 
   File temp = null; 
   for (int i = 0; i < file.length; i++) { 
    //如果oldPath以路径分隔符/或者\结尾,那么则oldPath/文件名就可以了 
    //否则要自己oldPath后面补个路径分隔符再加文件名 
    //谁知道你传递过来的参数是f:/a还是f:/a/啊? 
    if (oldPath.endsWith(File.separator)) { 
     temp = new File(oldPath + file[i]); 
    } else { 
     temp = new File(oldPath + File.separator + file[i]); 
    } 
     
    //如果游标遇到文件 
    if (temp.isFile()) { 
     FileInputStream input = new FileInputStream(temp); 
     FileOutputStream output = new FileOutputStream(newPath 
       + "/" + "rename_" + (temp.getName()).toString()); 
     byte[] bufferarray = new byte[1024 * 64]; 
     int prereadlength; 
     while ((prereadlength = input.read(bufferarray)) != -1) { 
      output.write(bufferarray, 0, prereadlength); 
     } 
     output.flush(); 
     output.close(); 
     input.close(); 
    } 
    //如果游标遇到文件夹 
    if (temp.isDirectory()) { 
     copyFolder(oldPath + "/" + file[i], newPath + "/" + file[i]); 
    } 
   } 
  } catch (Exception e) { 
   System.out.println("复制整个文件夹内容操作出错"); 
  } 
 } 
} 
 
public class FileCopy { 
 public static void main(String args[]) { 
  new FileTest().copyFolder("f:/a", "f:/b"); 
 
 } 
} 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。 

相关文章

  • 使用MultipartFile实现文件上传功能

    使用MultipartFile实现文件上传功能

    这篇文章主要介绍了使用MultipartFile实现文件上传功能,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-06-06
  • 如何使用IDEA创建MAPPER模板过程图解

    如何使用IDEA创建MAPPER模板过程图解

    这篇文章主要介绍了如何使用IDEA创建MAPPER模板,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-05-05
  • 从搭建Struts2 开发环境说起

    从搭建Struts2 开发环境说起

    本篇文章,小编为大家介绍从搭建Struts2 开发环境说起,有需要的朋友可以参考一下
    2013-04-04
  • 全面了解java基本类型和封装类型的区别及应用

    全面了解java基本类型和封装类型的区别及应用

    下面小编就为大家带来一篇全面了解java基本类型和封装类型的区别及应用。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-09-09
  • IntelliJ IDEA2020.1 Mac maven sdk 全局配置

    IntelliJ IDEA2020.1 Mac maven sdk 全局配置

    这篇文章主要介绍了IntelliJ IDEA2020.1 Mac maven sdk 全局配置,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • Struts2的输入校验实例代码

    Struts2的输入校验实例代码

    这篇文章主要介绍了Struts2的输入校验实例代码,非常不错,具有参考借鉴价值, 需要的朋友可以参考下
    2017-03-03
  • Java面向对象基础知识之枚举

    Java面向对象基础知识之枚举

    这篇文章主要介绍了Java面向对象的之枚举,文中有非常详细的代码示例,对正在学习java基础的小伙伴们有很好的帮助,需要的朋友可以参考下
    2021-11-11
  • 寻找二叉树最远的叶子结点(实例讲解)

    寻找二叉树最远的叶子结点(实例讲解)

    下面小编就为大家分享一篇寻找二叉树最远的叶子结点的实例讲解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-12-12
  • Feign如何实现第三方的HTTP请求

    Feign如何实现第三方的HTTP请求

    这篇文章主要介绍了Feign如何实现第三方的HTTP请求,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-10-10
  • IDEA下SpringBoot指定环境、配置文件启动操作过程

    IDEA下SpringBoot指定环境、配置文件启动操作过程

    这篇文章主要介绍了IDEA下SpringBoot指定环境、配置文件启动过程,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-08-08

最新评论