您好,登錄后才能下訂單哦!
這篇文章主要講解了“java io文件操作刪除文件或文件夾的方法”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“java io文件操作刪除文件或文件夾的方法”吧!
一、刪除文件或文件夾的四種基礎(chǔ)方法
下面的四個(gè)方法都可以刪除文件或文件夾,它們的共同點(diǎn)是:當(dāng)文件夾中包含子文件的時(shí)候都會(huì)刪除失敗,也就是說這四個(gè)方法只能刪除空文件夾。
需要注意的是:傳統(tǒng)IO中的File類和NIO中的Path類既可以代表文件,也可以代表文件夾。
File類的delete()File類的deleteOnExit()Files.delete(Path path)Files.deleteIfExists(Path path);
它們之間的差異:
File類的delete()true不能(返回false)不能(返回false)傳統(tǒng)IOFile類的deleteOnExit()void不能,但不存在就不會(huì)去執(zhí)行刪除不能(返回void)傳統(tǒng)IO,這是個(gè)坑,避免使用Files.delete(Path path)voidNoSuchFileExceptionDirectoryNotEmptyExceptionNIO,筆者推薦使用Files.deleteIfExists(Path path);truefalseDirectoryNotEmptyExceptionNIO由上面的對比可以看出,傳統(tǒng)IO方法刪除文件或文件夾,再刪除失敗的時(shí)候,最多返回一個(gè)false。通過這個(gè)false無法發(fā)掘刪除失敗的具體原因,是因?yàn)槲募旧聿淮嬖趧h除失敗?還是文件夾不為空導(dǎo)致的刪除失敗?NIO 的方法在這一點(diǎn)上,就做的比較好,刪除成功或失敗都有具體的返回值或者異常信息,這樣有利于我們在刪除文件或文件夾的時(shí)候更好的做程序的異常處理需要注意的是傳統(tǒng)IO中的deleteOnExit方法,筆者覺得應(yīng)該避免使用它。它永遠(yuǎn)只返回void,刪除失敗也不會(huì)有任何的Exception拋出,所以我建議不要用,以免在你刪除失敗的時(shí)候沒有任何的響應(yīng),而你可能誤以為刪除成功了。
//false只能告訴你失敗了 ,但是沒有給出任何失敗的原因@Testvoid testDeleteFileDir1() { File file = new File("D:\\data\\test"); boolean deleted = file.delete(); System.out.println(deleted);}//void ,刪除失敗沒有任何提示,應(yīng)避免使用這個(gè)方法,就是個(gè)坑@Testvoid testDeleteFileDir2() { File file = new File("D:\\data\\test1"); file.deleteOnExit();}//如果文件不存在,拋出NoSuchFileException//如果文件夾里面包含文件,拋出DirectoryNotEmptyException@Testvoid testDeleteFileDir3() throws IOException { Path path = Paths.get("D:\\data\\test1"); Files.delete(path); //返回值void}//如果文件不存在,返回false,表示刪除失敗(文件不存在)//如果文件夾里面包含文件,拋出DirectoryNotEmptyException@Testvoid testDeleteFileDir4() throws IOException { Path path = Paths.get("D:\\data\\test1"); boolean result = Files.deleteIfExists(path); System.out.println(result);}
歸根結(jié)底,建議大家使用java NIO的Files.delete(Path path)
和Files.deleteIfExists(Path path);
進(jìn)行文件或文件夾的刪除。
二、如何刪除整個(gè)目錄或者目錄中的部分文件
上文已經(jīng)說了,那四個(gè)API刪除文件夾的時(shí)候,如果文件夾包含子文件,就會(huì)刪除失敗。那么,如果我們確實(shí)想刪除整個(gè)文件夾,該怎么辦?
前提準(zhǔn)備
為了方便我們后面進(jìn)行試驗(yàn),先去創(chuàng)建這樣一個(gè)目錄結(jié)構(gòu),“.log”結(jié)尾的是數(shù)據(jù)文件,其他的是文件夾
可以使用代面的代碼進(jìn)行創(chuàng)建
private void createMoreFiles() throws IOException { Files.createDirectories(Paths.get("D:\\data\\test1\\test2\\test3\\test4\\test5\\")); Files.write(Paths.get("D:\\data\\test1\\test2\\test2.log"), "hello".getBytes()); Files.write(Paths.get("D:\\data\\test1\\test2\\test3\\test3.log"), "hello".getBytes());}
2.1. walkFileTree與FileVisitor 使用walkFileTree方法遍歷整個(gè)文件目錄樹,使用FileVisitor處理遍歷出來的每一項(xiàng)文件或文件夾FileVisitor的visitFile方法用來處理遍歷結(jié)果中的“文件”,所以我們可以在這個(gè)方法里面刪除文件FileVisitor的postVisitDirectory方法,注意方法中的“post”表示“后去做……”的意思,所以用來文件都處理完成之后再去處理文件夾,所以使用這個(gè)方法刪除文件夾就可以有效避免文件夾內(nèi)容不為空的異常,因?yàn)樵谌h除文件夾之前,該文件夾里面的文件已經(jīng)被刪除了。
@Testvoid testDeleteFileDir5() throws IOException { createMoreFiles(); Path path = Paths.get("D:\\data\\test1\\test2"); Files.walkFileTree(path, new SimpleFileVisitor<Path>() { // 先去遍歷刪除文件 @Override public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException { Files.delete(file); System.out.printf("文件被刪除 : %s%n", file); return FileVisitResult.CONTINUE; } // 再去遍歷刪除目錄 @Override public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException { Files.delete(dir); System.out.printf("文件夾被刪除: %s%n", dir); return FileVisitResult.CONTINUE; } } );}
下面的輸出體現(xiàn)了文件的刪除順序
文件被刪除 : D:\data\test1\test2\test2.log文件被刪除 : D:\data\test1\test2\test3\test3.log文件夾被刪除 : D:\data\test1\test2\test3\test4\test5文件夾被刪除 : D:\data\test1\test2\test3\test4文件夾被刪除 : D:\data\test1\test2\test3文件夾被刪除 : D:\data\test1\test2
我們既然可以遍歷出文件夾或者文件,我們就可以在處理的過程中進(jìn)行過濾。比如:
按文件名刪除文件或文件夾,參數(shù)Path里面含有文件或文件夾名稱按文件創(chuàng)建時(shí)間、修改時(shí)間、文件大小等信息去刪除文件,參數(shù)BasicFileAttributes 里面包含了這些文件信息。 2.2.Files.walk
如果你對Stream流語法不太熟悉的話,這種方法稍微難理解一點(diǎn),但是說實(shí)話也非常簡單。
使用Files.walk遍歷文件夾(包含子文件夾及子其文件),遍歷結(jié)果是一個(gè)Stream<Path>
對每一個(gè)遍歷出來的結(jié)果進(jìn)行處理,調(diào)用Files.delete就可以了。
@Testvoid testDeleteFileDir6() throws IOException { createMoreFiles(); Path path = Paths.get("D:\\data\\test1\\test2"); try (Stream<Path> walk = Files.walk(path)) { walk.sorted(Comparator.reverseOrder()) .forEach(DeleteFileDir::deleteDirectoryStream); }}private static void deleteDirectoryStream(Path path) { try { Files.delete(path); System.out.printf("刪除文件成功:%s%n",path.toString()); } catch (IOException e) { System.err.printf("無法刪除的路徑 %s%n%s", path, e); }}
問題:怎么能做到先去刪除文件,再去刪除文件夾? 。 利用的是字符串的排序規(guī)則,從字符串排序規(guī)則上講,“D:\data\test1\test2”一定排在“D:\data\test1\test2\test2.log”的前面。所以我們使用“sorted(Comparator.reverseOrder())”把Stream順序顛倒一下,就達(dá)到了先刪除文件,再刪除文件夾的目的。
下面的輸出,是最終執(zhí)行結(jié)果的刪除順序。
刪除文件成功:D:\data\test1\test2\test3\test4\test5刪除文件成功:D:\data\test1\test2\test3\test4刪除文件成功:D:\data\test1\test2\test3\test3.log刪除文件成功:D:\data\test1\test2\test3刪除文件成功:D:\data\test1\test2\test2.log刪除文件成功:D:\data\test1\test2
2.3.傳統(tǒng)IO-遞歸遍歷刪除文件夾
傳統(tǒng)的通過遞歸去刪除文件或文件夾的方法就比較經(jīng)典了
//傳統(tǒng)IO遞歸刪除@Testvoid testDeleteFileDir7() throws IOException { createMoreFiles(); File file = new File("D:\\data\\test1\\test2"); deleteDirectoryLegacyIO(file);}private void deleteDirectoryLegacyIO(File file) { File[] list = file.listFiles(); //無法做到list多層文件夾數(shù)據(jù) if (list != null) { for (File temp : list) { //先去遞歸刪除子文件夾及子文件 deleteDirectoryLegacyIO(temp); //注意這里是遞歸調(diào)用 } } if (file.delete()) { //再刪除自己本身的文件夾 System.out.printf("刪除成功 : %s%n", file); } else { System.err.printf("刪除失敗 : %s%n", file); }}
需要注意的是:
listFiles()
方法只能列出文件夾下面的一層文件或文件夾,不能列出子文件夾及其子文件。先去遞歸刪除子文件夾,再去刪除文件夾自己本身
感謝各位的閱讀,以上就是“java io文件操作刪除文件或文件夾的方法”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對java io文件操作刪除文件或文件夾的方法這一問題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識點(diǎn)的文章,歡迎關(guān)注!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。