java?io文件操作刪除文件或文件夾的7種方法
本文是Java IO總結(jié)系列篇的第4篇,前篇的訪問(wèn)地址如下:
總結(jié)java中創(chuàng)建并寫文件的5種方式-JAVA IO基礎(chǔ)總結(jié)第一篇
總結(jié)java從文件中讀取數(shù)據(jù)的6種方法-JAVA IO基礎(chǔ)總結(jié)第二篇
總結(jié)java創(chuàng)建文件夾的4種方法及其優(yōu)缺點(diǎn)-JAVA IO基礎(chǔ)總結(jié)第三篇
為了方便大家理解,我特意制作了本文對(duì)應(yīng)的視頻:總結(jié)刪除文件或者文件夾的7種方法
一、刪除文件或文件夾的四種基礎(chǔ)方法
下面的四個(gè)方法都可以刪除文件或文件夾,它們的共同點(diǎn)是:當(dāng)文件夾中包含子文件的時(shí)候都會(huì)刪除失敗,也就是說(shuō)這四個(gè)方法只能刪除空文件夾。
需要注意的是:傳統(tǒng)IO中的File類和NIO中的Path類既可以代表文件,也可以代表文件夾。
File類的delete()
File類的deleteOnExit()
Files.delete(Path path)
Files.deleteIfExists(Path path);
它們之間的差異:
成功的返回值 | 是否能判別文件夾不存在導(dǎo)致失敗 | 是否能判別文件夾不為空導(dǎo)致失敗 | 備注 | |
---|---|---|---|---|
File類的delete() | true | 不能(返回false) | 不能(返回false) | 傳統(tǒng)IO |
File類的deleteOnExit() | void | 不能,但不存在就不會(huì)去執(zhí)行刪除 | 不能(返回void) | 傳統(tǒng)IO,這是個(gè)坑,避免使用 |
Files.delete(Path path) | void | NoSuchFileException | DirectoryNotEmptyException | NIO,筆者推薦使用 |
Files.deleteIfExists(Path path); | true | false | DirectoryNotEmptyException | NIO |
由上面的對(duì)比可以看出,傳統(tǒng)IO方法刪除文件或文件夾,再刪除失敗的時(shí)候,最多返回一個(gè)false。通過(guò)這個(gè)false無(wú)法發(fā)掘刪除失敗的具體原因,是因?yàn)槲募旧聿淮嬖趧h除失???還是文件夾不為空導(dǎo)致的刪除失???
NIO 的方法在這一點(diǎn)上,就做的比較好,刪除成功或失敗都有具體的返回值或者異常信息,這樣有利于我們?cè)趧h除文件或文件夾的時(shí)候更好的做程序的異常處理
需要注意的是傳統(tǒng)IO中的deleteOnExit方法,筆者覺(jué)得應(yīng)該避免使用它。它永遠(yuǎn)只返回void,刪除失敗也不會(huì)有任何的Exception拋出,所以我建議不要用,以免在你刪除失敗的時(shí)候沒(méi)有任何的響應(yīng),而你可能誤以為刪除成功了。
//false只能告訴你失敗了 ,但是沒(méi)有給出任何失敗的原因 @Test void testDeleteFileDir1() { File file = new File("D:\\data\\test"); boolean deleted = file.delete(); System.out.println(deleted); } //void ,刪除失敗沒(méi)有任何提示,應(yīng)避免使用這個(gè)方法,就是個(gè)坑 @Test void testDeleteFileDir2() { File file = new File("D:\\data\\test1"); file.deleteOnExit(); } //如果文件不存在,拋出NoSuchFileException //如果文件夾里面包含文件,拋出DirectoryNotEmptyException @Test void testDeleteFileDir3() throws IOException { Path path = Paths.get("D:\\data\\test1"); Files.delete(path); //返回值void } //如果文件不存在,返回false,表示刪除失敗(文件不存在) //如果文件夾里面包含文件,拋出DirectoryNotEmptyException @Test void 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)說(shuō)了,那四個(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è)文件目錄樹(shù),使用FileVisitor處理遍歷出來(lái)的每一項(xiàng)文件或文件夾
FileVisitor的visitFile方法用來(lái)處理遍歷結(jié)果中的“文件”,所以我們可以在這個(gè)方法里面刪除文件
FileVisitor的postVisitDirectory方法,注意方法中的“post”表示“后去做……”的意思,所以用來(lái)文件都處理完成之后再去處理文件夾,所以使用這個(gè)方法刪除文件夾就可以有效避免文件夾內(nèi)容不為空的異常,因?yàn)樵谌h除文件夾之前,該文件夾里面的文件已經(jīng)被刪除了。
@Test void 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
我們既然可以遍歷出文件夾或者文件,我們就可以在處理的過(guò)程中進(jìn)行過(guò)濾。比如:
- 按文件名刪除文件或文件夾,參數(shù)Path里面含有文件或文件夾名稱
- 按文件創(chuàng)建時(shí)間、修改時(shí)間、文件大小等信息去刪除文件,參數(shù)BasicFileAttributes 里面包含了這些文件信息。
2.2.Files.walk
如果你對(duì)Stream流語(yǔ)法不太熟悉的話,這種方法稍微難理解一點(diǎn),但是說(shuō)實(shí)話也非常簡(jiǎn)單。
使用Files.walk遍歷文件夾(包含子文件夾及子其文件),遍歷結(jié)果是一個(gè)Stream<Path>
對(duì)每一個(gè)遍歷出來(lái)的結(jié)果進(jìn)行處理,調(diào)用Files.delete就可以了。
@Test void 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("無(wú)法刪除的路徑 %s%n%s", path, e); } }
問(wèn)題:怎么能做到先去刪除文件,再去刪除文件夾? 。 利用的是字符串的排序規(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)的通過(guò)遞歸去刪除文件或文件夾的方法就比較經(jīng)典了
//傳統(tǒng)IO遞歸刪除 @Test void testDeleteFileDir7() throws IOException { createMoreFiles(); File file = new File("D:\\data\\test1\\test2"); deleteDirectoryLegacyIO(file); } private void deleteDirectoryLegacyIO(File file) { File[] list = file.listFiles(); //無(wú)法做到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文件操作刪除文件或文件夾的7種方法的詳細(xì)內(nèi)容,更多關(guān)于java io文件操作刪除文件或文件夾的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
校驗(yàn)非空的注解@NotNull如何取得自定義的message
這篇文章主要介紹了校驗(yàn)非空的注解@NotNull如何取得自定義的message,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09java隨機(jī)數(shù)生成具體實(shí)現(xiàn)代碼
這篇文章主要為大家分享了java隨機(jī)數(shù)生成具體實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-04-04SpringBoot靜態(tài)資源與首頁(yè)配置實(shí)現(xiàn)原理深入分析
最近在做SpringBoot項(xiàng)目的時(shí)候遇到了“白頁(yè)”問(wèn)題,通過(guò)查資料對(duì)SpringBoot訪問(wèn)靜態(tài)資源做了總結(jié),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2022-10-10java調(diào)用FFmpeg實(shí)現(xiàn)視屏壓縮功能的詳細(xì)步驟
這篇文章主要介紹了java調(diào)用FFmpeg實(shí)現(xiàn)視屏壓縮功能,本文簡(jiǎn)單的展示了java調(diào)用FFmpeg命令實(shí)現(xiàn)視屏的壓縮的詳細(xì)步驟,需要的朋友可以參考下2021-09-09使用SpringBoot打jar包并部署到Tomcat詳細(xì)步驟
今天帶大家來(lái)學(xué)習(xí)怎么使用SpringBoot打jar包并部署到Tomcat,文中有非常詳細(xì)的步驟及代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們很有幫助,需要的朋友可以參考下2021-05-05SpringBoot啟動(dòng)流程SpringApplication準(zhǔn)備階段源碼分析
這篇文章主要為大家介紹了SpringBoot啟動(dòng)流程SpringApplication準(zhǔn)備階段源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04springboot集成activemq的實(shí)例代碼
本篇文章主要介紹了springboot集成activemq的實(shí)例代碼,詳細(xì)的介紹了ActiveMQ和Spring-Boot 集成 ActiveMQ,有興趣的可以了解下。2017-05-05