一、基本目标
使用java完成如下的操作:
把一个文件夹内的所有文件拷贝的另一的文件夹,例如,在f盘中有a与b两个文件夹:
f:/a里面有一堆文件,运行java程序之后就会全部复制到f:/b,并且完成重命名,在所有文件之前加rename_的前缀,如果里面有文件夹,则文件夹不重命名,里面的文件进行重命名,同样在所有文件之前加rename_的前缀:
二、制作过程
1、首先主函数非常简单,就是调用了上面filetest类中的copyfolder函数
1
2
3
4
5
6
|
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.*;由于用到了输入输出流
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
|
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、因此整个程序如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
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" ); } } |
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:http://blog.csdn.net/yongh701/article/details/42964557