在程序中對(duì)文件進(jìn)行壓縮解壓縮是很重要的功能,不僅能減小文件的體積,還能對(duì)文件起到保護(hù)作用。如果是生成用戶可以下載的文件,還可以極大的減少網(wǎng)絡(luò)流量并提升下載速度。最近在一個(gè) C# 項(xiàng)目中用到了創(chuàng)建壓縮文件的功能,在此和同學(xué)們分享一下使用心得。
SharpZipLib 庫
既然是很重要的用能,那么如果每個(gè)人在使用的時(shí)候都去用基本的 API 去實(shí)現(xiàn)一遍顯然不符合效率至上的生產(chǎn)要求。作為比較有經(jīng)驗(yàn)的開發(fā)人員相信您一定會(huì)在第一時(shí)間去搜尋一款功能豐富,口碑良好的開源類庫來完成相關(guān)的工作。在 .NET 平臺(tái)上,要操作壓縮文件的話您的第一選擇一定是 SharpZipLib。SharpZipLib 是一個(gè)開源的基于 .NET 平臺(tái)的壓縮、解壓縮類庫。特點(diǎn)是經(jīng)過長(zhǎng)期的開發(fā)和使用現(xiàn)在已經(jīng)變得非常的穩(wěn)定,可以放心的應(yīng)用到產(chǎn)品中。下面我們就通過實(shí)例來介紹如何使用它在 C# 代碼中創(chuàng)建壓縮文件,以及一些常見問題的處理方法。SharpZipLib 的下載請(qǐng)?jiān)L問這里。編譯也很簡(jiǎn)單,用 VisualStudio 打開直接編譯就能成功。如果您想全面的掌握 SharpZipLib 的使用方法,建議您直接去讀 SharpZipLib 的文檔,本文僅介紹基本的用法和一些使用心得。
基本壓縮操作
SharpZipLib 支持 Zip,Gzip,Tar,BZip2 等主流的壓縮格式。本文以 zip 格式做介紹,其它格式的用法也都差不太多。對(duì)于 zip 壓縮格式,創(chuàng)建壓縮文件時(shí)用到的類型主要為 ZipOutputStream 和 ZipEntry。下面通過幾個(gè)典型的用例來介紹它們的用法。
讀取硬盤上的文件并加入壓縮包
這可能是最簡(jiǎn)單也最常見的用法了,直接上代碼:
//生成的壓縮文件為test.zip using (FileStream fsOut = File.Create("test.zip")) { //ZipOutputStream類的構(gòu)造函數(shù)需要一個(gè)流,文件流、內(nèi)存流都可以,壓縮后的內(nèi)容會(huì)寫入到這個(gè)流中。 using (ZipOutputStream zipStream = new ZipOutputStream(fsOut)) { //準(zhǔn)備把G盤根目錄下的vcredist_x86.exe文件添加到壓縮包中。 string fileName = @"G:\vcredist_x86.exe"; FileInfo fi = new FileInfo(fileName); //entryName就是壓縮包中文件的名稱。 string entryName = "vcredist_x86.exe"; //ZipEntry類代表了一個(gè)壓縮包中的一個(gè)項(xiàng),可以是一個(gè)文件,也可以是一個(gè)目錄。 ZipEntry newEntry = new ZipEntry(entryName); newEntry.DateTime = fi.LastWriteTime; newEntry.Size = fi.Length; //把壓縮項(xiàng)的信息添加到ZipOutputStream中。 zipStream.PutNextEntry(newEntry); byte[] buffer = new byte[4096]; //把需要壓縮文件以文件流的方式復(fù)制到ZipOutputStream中。 using (FileStream streamReader = File.OpenRead(fileName)) { StreamUtils.Copy(streamReader, zipStream, buffer); } zipStream.CloseEntry(); //添加多個(gè)文件 //如果要壓縮一個(gè)文件夾,就是通過遍歷添加文件夾下所有的文件 string fileName2 = @"G:\share\web.dll"; FileInfo fi2 = new FileInfo(fileName2); //文件在壓縮包中的路徑 string entryName2 = "share\\web.dll"; ZipEntry newEntry2 = new ZipEntry(entryName2); newEntry2.DateTime = fi2.LastWriteTime; newEntry2.Size = fi2.Length; zipStream.PutNextEntry(newEntry2); byte[] buffer2 = new byte[4096]; using (FileStream streamReader = File.OpenRead(fileName2)) { StreamUtils.Copy(streamReader, zipStream, buffer2); } zipStream.CloseEntry(); //使用流操作時(shí)一定要設(shè)置IsStreamOwner為false。否則很容易發(fā)生在文件流關(guān)閉后的異常。 zipStream.IsStreamOwner = false; zipStream.Finish(); zipStream.Close(); } }
代碼并不復(fù)雜且添加了詳細(xì)的注釋,因此不再贅言。此時(shí)已經(jīng)完成了把文件加入壓縮包的功能,壓縮包中的內(nèi)容如下:

注意,web.dll 文件在 share 文件夾中。
把內(nèi)存中的數(shù)據(jù)添加到壓縮包
有時(shí)我們要壓縮的對(duì)象并不是磁盤上的文件,而是內(nèi)存中的數(shù)據(jù)。比如數(shù)據(jù)庫查詢操作的結(jié)果中有一些字符串,希望把這些字符串寫入到壓縮包中的文本文件中。當(dāng)然可以先把這些字符串保存到磁盤上的文件中,然后再通過前面例子中的方法寫入壓縮包,這也可以完成任務(wù),卻不是高效的方法。首先磁盤 IO 很慢也很昂貴,另外在一些 web 應(yīng)用環(huán)境中你是沒有權(quán)限寫文件的。這就要求我們直接把數(shù)據(jù)寫入到壓縮包中:
//我們有一個(gè)字符串,希望直接寫入到壓縮包中的City.csv文件中。 byte[] string1 = Encoding.UTF8.GetBytes("Washington,ShangHai,TianJin,DongJing"); using (FileStream fsOut = File.Create("test1.zip")) { using (ZipOutputStream zipStream = new ZipOutputStream(fsOut)) { ZipEntry entry = new ZipEntry("City.csv"); entry.DateTime = DateTime.Now; zipStream.PutNextEntry(entry); //Write方法和前面用的StreamUtils.Copy方法差不多,不過這里操作的是byte數(shù)組。 zipStream.Write(string1, 0, string1.Length); zipStream.CloseEntry(); zipStream.IsStreamOwner = false; zipStream.Finish(); zipStream.Close(); } }
這次我們把內(nèi)存中的一個(gè)字符串直接寫入了壓縮包中得 City.csv 文件。看上去還不錯(cuò),至少代碼看上去還算清爽。接下來看看我們還能干些什么?
把壓縮包保存在內(nèi)存中