Java組件開發(fā)之文件壓縮與解壓詳解
緣起
在服務(wù)端項(xiàng)目開發(fā)的過程中,需要要各種不同的數(shù)據(jù)打交道,有時(shí)為了方便用戶上傳的下載會將數(shù)據(jù)以壓縮包的形式打包后進(jìn)行傳輸,那么就會遇到壓縮與解壓的問題了。在現(xiàn)在擁有如此多的開源組件的情況下,壓縮本身并不算一個(gè)復(fù)雜的任務(wù),只需要找到一些合適的maven庫即可解決咱們的問題。那么為什么我們還需要做一個(gè)處理壓縮的公共組件呢? 不妨我們想象一下這樣一些需求:
- 壓縮包中包含很多種不同的文件,我們只需要解壓其中的部分類型的數(shù)據(jù);
- 待壓縮的目錄中只有滿足條件的文件需要被添加到壓縮包中;
- 壓縮格式類型多樣,需要針對每種都單獨(dú)適配;
- ...
如果分析這些需求就會發(fā)現(xiàn)這些都是通用功能,如果封裝好了,業(yè)務(wù)方只需要簡單調(diào)用即可,省去了很多重復(fù)的功能開發(fā)。
設(shè)計(jì)
通過分析上面的需求可知,需求大致分為兩大類:
- 文件過濾;
- 兼容不同的壓縮文件類型;
好了,那么我們可以繼續(xù)的針對需求來實(shí)現(xiàn)咱們的組件了。
如何處理文件過濾的功能呢?對于一個(gè)公共組件而言,它是沒辦法聰明到了解所有的業(yè)務(wù)方的需求的,也沒有必要了解,因?yàn)樾枨笞兓窈R粯?,哪里能做到面面俱到呢,你們說是吧?既然如此,那就只能是將過濾的功能暴露給業(yè)務(wù)方,組件里面接收過濾后的結(jié)果后進(jìn)行相應(yīng)處理即可,恩...完美!那么如何來實(shí)現(xiàn)呢?相信聰明的你可能想到了java8里的文件遍歷Files.walkFileTree()。是的,咱們想到了一塊。咱們可以仿照這個(gè)寫法,讓業(yè)務(wù)方傳入FileVisitor的實(shí)現(xiàn)即可,這樣就可以在調(diào)用方以最少的認(rèn)知的情況下了解如何來使用這個(gè)公共組件。這樣后面推動的時(shí)候就更容易啦,你說是吧?
上代碼:
/** * 壓縮示例,目錄下的文件全部壓縮,不需要過濾 */ Compressions.compress(new File(parent), new File(parent, "test.zip"));
/** * 解壓示例,全部解壓,不需要過濾 */ Compressions.uncompress(new File(parent,"test.zip"), new File(parent,"test"), true);
/** * 壓縮示例,按需求過濾 */ Compressions.compress(new File(parent), new File(parent, "test.zip"), new FileVisitor<Path>() { @Override public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException { if (dir.getFileName().toString().equals("testq")) { return FileVisitResult.SKIP_SUBTREE; } return FileVisitResult.CONTINUE; } @Override public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException { if (file.getFileName().toString().endsWith("zip")) { return FileVisitResult.TERMINATE; } return FileVisitResult.CONTINUE; } @Override public FileVisitResult visitFileFailed(Path file, IOException exc) throws IOException { return FileVisitResult.CONTINUE; } @Override public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException { return FileVisitResult.CONTINUE; } });
/** * 解壓示例,按需求過濾 */ Compressions.uncompress(new File(path), new SimpleFileVisitor<>() { @Override public FileVisitResult preVisitDirectory(FileEntry dir, BasicFileAttributes attrs) throws IOException { File file = new File(parent, dir.getPath()); if (!file.exists()) { file.mkdirs(); } return super.preVisitDirectory(dir, attrs); } @Override public FileVisitResult visitFile(FileEntry fileEntry, BasicFileAttributes attrs) throws IOException { System.out.println("visitFile:" + fileEntry.getPath()); if (fileEntry.getPath().endsWith("jpg")){ File file = new File(parent, fileEntry.getPath()); fileEntry.transferTo(file); } return super.visitFile(fileEntry, attrs); } @Override public FileVisitResult visitFileFailed(FileEntry<?> file, IOException exc) throws IOException { //讀取出異常的壓縮包全路徑 String path2 = file.getPath(); return FileVisitResult.CONTINUE; } });
怎么樣,使用起來是不是很方便,很靈活。
文件過濾的問題解決了,那么咱們再進(jìn)一步,兼容下比較常用的壓縮文件類型。 自己寫是不可能自己去寫的了,別忘了咱們是調(diào)用工程師。為了支持zip/7z/rar/jar/tar/tar.gz等常用的格式,咱們得導(dǎo)入幾個(gè)開源庫:
- net.sf.sevenzipjbinding (zip/7z/rar/jar/tar);
- commons-io (tar.gz)
插個(gè)題外話,這里要重點(diǎn)表揚(yáng)下net.sf.sevenzipjbinding 在解壓縮文件過濾上擁有顯著的性能優(yōu)勢,性能優(yōu)勢從何而來呢,這里就要講到net.sf.sevenzipjbinding和commons-io是如何處理需要跳過的文件流了,net.sf.sevenzipjbinding是真的跳過,就像RandomAccessFile的seek功能,而commons-io則是將跳過的文件流完整讀一遍,但不作處理,哪個(gè)性能會更好,聰明的你肯定已經(jīng)有了答案。事實(shí)上我也考慮過把commons-io中讀流的操作改為跳過,技術(shù)上是可實(shí)現(xiàn)的,奈何我比較懶...思路大概是用RandomAccessFile或者SeekableByteChannel來實(shí)現(xiàn)Seek跳轉(zhuǎn),感覺RandomAccessFile可能會更容易些,畢竟跟它里面使用的api是一個(gè)時(shí)代的東東。
好了,咱們已經(jīng)兼容了好幾種不同的壓縮格式,那么我們就可以根據(jù)不同的文件類型來自動使用匹配的壓縮處理類。調(diào)用方可以做到完全無感,就很nice!當(dāng)然我現(xiàn)在偷懶,是通過文件的擴(kuò)展名來匹配的,更好的辦法應(yīng)該是通過文件的元信息來匹配,比如魔數(shù)。沒辦法,我就是懶,而且目前用起來確實(shí)夠用了,哈哈。
到此這篇關(guān)于Java組件開發(fā)之文件壓縮與解壓詳解的文章就介紹到這了,更多相關(guān)Java文件壓縮與解壓內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringSecurity授權(quán)實(shí)現(xiàn)基本思路
本文介紹了SpringSecurity中使用FilterSecurityInterceptor進(jìn)行權(quán)限校驗(yàn)的基本方法,通過SecurityContextHolder獲取Authentication中的權(quán)限信息,感興趣的朋友跟隨小編一起看看吧2024-10-10logback TimeBasedRollingPolicy按天生成日志源碼解析
這篇文章主要為大家介紹了logback TimeBasedRollingPolicy按天生成日志源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11log4j2 項(xiàng)目日志組件的實(shí)例代碼
下面小編就為大家分享一篇log4j2 項(xiàng)目日志組件的實(shí)例代碼,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2017-12-12Java基本概念監(jiān)視器實(shí)習(xí)原理解析
這篇文章主要介紹了Java基本概念監(jiān)視器實(shí)習(xí)原理解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08Java實(shí)戰(zhàn)練習(xí)之撲克牌魔術(shù)
這篇文章主要介紹了Java實(shí)戰(zhàn)練習(xí)之撲克牌魔術(shù),文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)java的小伙伴們有很好地幫助,需要的朋友可以參考下2021-04-04Spring項(xiàng)目中使用Junit單元測試并配置數(shù)據(jù)源的操作
這篇文章主要介紹了Spring項(xiàng)目中使用Junit單元測試并配置數(shù)據(jù)源的操作,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09