Java利用文件輸入輸出流實(shí)現(xiàn)文件夾內(nèi)所有文件拷貝到另一個(gè)文件夾
一、基本目標(biāo)
使用Java完成如下的操作:
把一個(gè)文件夾內(nèi)的所有文件拷貝的另一的文件夾,例如,在F盤中有a與b兩個(gè)文件夾:

f:/a里面有一堆文件,運(yùn)行Java程序之后就會(huì)全部復(fù)制到f:/b,并且完成重命名,在所有文件之前加rename_的前綴,如果里面有文件夾,則文件夾不重命名,里面的文件進(jìn)行重命名,同樣在所有文件之前加rename_的前綴:

二、制作過(guò)程
1、首先主函數(shù)非常簡(jiǎn)單,就是調(diào)用了上面FileTest類中的copyFolder函數(shù)
public class FileCopy {
public static void main(String args[]) {
new FileTest().copyFolder("f:/a", "f:/b");
}
}
值得注意的是,這個(gè)的傳遞過(guò)去的參數(shù)的路徑寫法,在java中,f:/a是沒(méi)有問(wèn)題的,f:\a也是沒(méi)有問(wèn)題的,但是由于\在字符串表達(dá)的時(shí)候,必須轉(zhuǎn)移,所以你必須寫成f:\\a
2、整個(gè)程序的關(guān)鍵在這個(gè)FileTest類中的copyFolder函數(shù),這個(gè)類里面就這個(gè)函數(shù)-_-!而且注意在程序開頭引入java.io.*;由于用到了輸入輸出流
class FileTest {
public void copyFolder(String oldPath, String newPath) {
try {
// 如果文件夾不存在,則建立新文件夾
(new File(newPath)).mkdirs();
//讀取整個(gè)文件夾的內(nèi)容到file字符串?dāng)?shù)組,下面設(shè)置一個(gè)游標(biāo)i,不停地向下移開始讀這個(gè)數(shù)組
File filelist = new File(oldPath);
String[] file = filelist.list();
//要注意,這個(gè)temp僅僅是一個(gè)臨時(shí)文件指針
//整個(gè)程序并沒(méi)有創(chuàng)建臨時(shí)文件
File temp = null;
for (int i = 0; i < file.length; i++) {
//如果oldPath以路徑分隔符/或者\(yùn)結(jié)尾,那么則oldPath/文件名就可以了
//否則要自己oldPath后面補(bǔ)個(gè)路徑分隔符再加文件名
//誰(shuí)知道你傳遞過(guò)來(lái)的參數(shù)是f:/a還是f:/a/?。?
if (oldPath.endsWith(File.separator)) {
temp = new File(oldPath + file[i]);
} else {
temp = new File(oldPath + File.separator + file[i]);
}
//如果游標(biāo)遇到文件
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();
}
//如果游標(biāo)遇到文件夾
if (temp.isDirectory()) {
copyFolder(oldPath + "/" + file[i], newPath + "/" + file[i]);
}
}
} catch (Exception e) {
System.out.println("復(fù)制整個(gè)文件夾內(nèi)容操作出錯(cuò)");
}
}
}
可能游標(biāo)遇到文件部分有點(diǎn)難以理解,其實(shí)是這樣的,首先設(shè)置一個(gè)文件的輸入流,指定從游標(biāo)遇到的文件中輸入,再指定輸出到newPath/rename_舊文件的文件名這個(gè)文件目錄,之后,設(shè)置一個(gè)緩沖數(shù)組,文件輸入流對(duì)于自己要讀取的文件,每次調(diào)用read方法,它都會(huì)向后繼續(xù)上一次讀取的位置繼續(xù)讀取緩沖數(shù)組bufferarray的長(zhǎng)度的內(nèi)容,把讀取到的內(nèi)容存儲(chǔ)到緩沖數(shù)組,覆蓋緩沖數(shù)組之前的所有內(nèi)容,然后文件輸出流會(huì)把緩沖數(shù)組的所有內(nèi)容輸出的指定的位置,直到文件輸入流遇到了-1。
至于文件輸入流為何能這樣按順序,每次都會(huì)向后繼續(xù)上一次讀取的位置繼續(xù)讀取,那是因?yàn)楫?dāng)要進(jìn)行文件的讀取,Java封裝的FileInputStream.read方法也會(huì)調(diào)用操作系統(tǒng)的API依次讀取這些數(shù)據(jù)。在讀取文件數(shù)據(jù)的時(shí)候必須是順序的,不可能說(shuō)先讀取第一個(gè)字節(jié),后讀取倒數(shù)第二個(gè)字節(jié)。循環(huán)讀取的時(shí)候就read方法將讀取的位置++,因此造成每次read都是順序讀取后面的字節(jié),直到遇到文件末尾標(biāo)記。
當(dāng)游標(biāo)遇到文件夾則重新調(diào)用自己完成同樣的操作即可,這就是所謂的迭代。
3、因此整個(gè)程序如下:
import java.io.*;
/**
*
* @param oldPath 被拷貝的目錄
* @param newPath 要拷貝到的目錄
*
*/
class FileTest {
public void copyFolder(String oldPath, String newPath) {
try {
// 如果文件夾不存在,則建立新文件夾
(new File(newPath)).mkdirs();
//讀取整個(gè)文件夾的內(nèi)容到file字符串?dāng)?shù)組,下面設(shè)置一個(gè)游標(biāo)i,不停地向下移開始讀這個(gè)數(shù)組
File filelist = new File(oldPath);
String[] file = filelist.list();
//要注意,這個(gè)temp僅僅是一個(gè)臨時(shí)文件指針
//整個(gè)程序并沒(méi)有創(chuàng)建臨時(shí)文件
File temp = null;
for (int i = 0; i < file.length; i++) {
//如果oldPath以路徑分隔符/或者\(yùn)結(jié)尾,那么則oldPath/文件名就可以了
//否則要自己oldPath后面補(bǔ)個(gè)路徑分隔符再加文件名
//誰(shuí)知道你傳遞過(guò)來(lái)的參數(shù)是f:/a還是f:/a/啊?
if (oldPath.endsWith(File.separator)) {
temp = new File(oldPath + file[i]);
} else {
temp = new File(oldPath + File.separator + file[i]);
}
//如果游標(biāo)遇到文件
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();
}
//如果游標(biāo)遇到文件夾
if (temp.isDirectory()) {
copyFolder(oldPath + "/" + file[i], newPath + "/" + file[i]);
}
}
} catch (Exception e) {
System.out.println("復(fù)制整個(gè)文件夾內(nèi)容操作出錯(cuò)");
}
}
}
public class FileCopy {
public static void main(String args[]) {
new FileTest().copyFolder("f:/a", "f:/b");
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Idea?編譯并運(yùn)行?Spark?3.1.1?源碼的方法
這篇文章主要介紹了Idea?編譯并運(yùn)行?Spark?3.1.1源碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-11-11
Springboot如何通過(guò)filter修改Header的值
這篇文章主要介紹了Springboot如何通過(guò)filter修改Header的值問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07
Tomcat?8.5?+mysql?5.7+jdk1.8開發(fā)JavaSE的金牌榜小項(xiàng)目
這篇文章主要介紹了Tomcat?8.5?+mysql?5.7+jdk1.8開發(fā)JavaSE的金牌榜小項(xiàng)目,本文通過(guò)圖文實(shí)例相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
Java+opencv3.2.0實(shí)現(xiàn)hough直線檢測(cè)
這篇文章主要為大家詳細(xì)介紹了Java+opencv3.2.0之hough直線檢測(cè),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-02-02

