欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

c語言壓縮文件詳細(xì)講解

 更新時(shí)間:2021年09月07日 15:51:52   作者:micDavid  
這篇文章主要從單文件壓縮、多文件壓縮、多文件異步壓縮講訴了c語言壓縮文件,需要的朋友可以參考下面具體的文章內(nèi)容

c語言壓縮文件

話說當(dāng)今壓縮市場三足鼎立,能叫上名號的有zip、rar、7z。其中zip是壓縮界的鼻祖,在各大平臺上的流行度最廣,rar是商業(yè)軟件,壓縮率和效率都是很高的,對個(gè)人用戶沒有限制。7z是開源的,屬于后起之秀,也有著不凡的壓縮率,但在內(nèi)存占有率的問題上。今天,主要總結(jié)下,windows平臺下,zip的壓縮與解壓的方法,用ICSharpCode組件。

一、單文件壓縮

      場景,文件可能比較大,需要壓縮傳輸,比如上傳和下載

        /// <summary>
         /// 單文件壓縮
        /// </summary>
          /// <param name="sourceFile">源文件</param>
          /// <param name="zipedFile">zip壓縮文件</param>
          /// <param name="blockSize">緩沖區(qū)大小</param>
          /// <param name="compressionLevel">壓縮級別</param>
         public static void ZipFile(string sourceFile, string zipedFile, int blockSize = 1024, int compressionLevel = 6)
         {
             if (!File.Exists(sourceFile))
            {
                throw new System.IO.FileNotFoundException("The specified file " + sourceFile + " could not be found.");
             }
             var fileName = System.IO.Path.GetFileNameWithoutExtension(sourceFile);
 
            FileStream streamToZip = new FileStream(sourceFile, FileMode.Open, FileAccess.Read);
             FileStream zipFile = File.Create(zipedFile);
            ZipOutputStream zipStream = new ZipOutputStream(zipFile);
 
             ZipEntry zipEntry = new ZipEntry(fileName);
             zipStream.PutNextEntry(zipEntry);
 
            //存儲、最快、較快、標(biāo)準(zhǔn)、較好、最好  0-9
           zipStream.SetLevel(compressionLevel);
 
             byte[] buffer = new byte[blockSize];
 
             int size = streamToZip.Read(buffer, 0, buffer.Length);
             zipStream.Write(buffer, 0, size);
             try
             {
                 while (size < streamToZip.Length)
                 {
                   int sizeRead = streamToZip.Read(buffer, 0, buffer.Length);
                    zipStream.Write(buffer, 0, sizeRead);
                    size += sizeRead;
                }
            }
             catch (Exception ex)
             {
                throw ex;
             }
            zipStream.Finish();
            zipStream.Close();
             streamToZip.Close();
         }


說明:26行,blocksize為緩存區(qū)大小,不能設(shè)置太大,如果太大也會報(bào)異常。26-38行,把文件通過FileStream流,讀取到緩沖區(qū)中,再寫入到ZipOutputStream流。你可以想象,兩個(gè)管道,一個(gè)讀,另一個(gè)寫,中間是緩沖區(qū),它們的工作方式是同步的方式。想一下,能不能以異步的方式工作,讀的管道只管讀,寫的管道只管寫?如果是這樣一個(gè)場景,讀的特別快,寫的比較慢,比如,不是本地寫,而是要經(jīng)過網(wǎng)絡(luò)傳輸,就可以考慮異步的方式。怎么做,讀者可以自行改造。關(guān)鍵一點(diǎn),流是有順序的,所以要保證順序的正確性即可。

二、多文件壓縮

      這種場景也是比較多見,和單文件壓縮類似,無非就是多循環(huán)幾次。

          /// <summary>
          /// 多文件壓縮
         /// </summary>
         /// <param name="zipfile">zip壓縮文件</param>
         /// <param name="filenames">源文件集合</param>
         /// <param name="password">壓縮加密</param>
         public void ZipFiles(string zipfile, string[] filenames, string password = "")
         {
              ZipOutputStream s = new ZipOutputStream(System.IO.File.Create(zipfile));
 
            s.SetLevel(6);
              if (password != "")
                 s.Password = Md5Help.Encrypt(password);
 
             foreach (string file in filenames)
             {
                //打開壓縮文件 
                FileStream fs = File.OpenRead(file);
 
                byte[] buffer = new byte[fs.Length];
                 fs.Read(buffer, 0, buffer.Length);
 
                 var name = Path.GetFileName(file);

                 ZipEntry entry = new ZipEntry(name);
                 entry.DateTime = DateTime.Now;
                 entry.Size = fs.Length;
                fs.Close();
                 s.PutNextEntry(entry);
                 s.Write(buffer, 0, buffer.Length);
             }
             s.Finish();
             s.Close();
         }

說明:21行,緩沖區(qū)大小直接為文件大小,所以一次讀完,沒有循環(huán)讀寫。這種情況下,單個(gè)文件不能太大,比如超過1G。14行,可以為壓縮包設(shè)置密碼,

MD5的生成方法如下:

    public class Md5Help
    {
        /// <summary>
        ///32位 MD5加密
        /// </summary>
        /// <param name="str">加密字符</param>
        /// <returns></returns>
        public static string Encrypt(string str)
        {
            MD5 md5 = new MD5CryptoServiceProvider();

            byte[] encryptdata = md5.ComputeHash(Encoding.UTF8.GetBytes(str));

            return Convert.ToBase64String(encryptdata);
        }
    }

 

三、多文件異步壓縮

      上面同步的壓縮的前提是,假設(shè)文件不大,而且文件數(shù)不多,但是現(xiàn)實(shí)是,不光文件大,而且文件數(shù)比較多。這種情況,就要考慮異步方法了。否則會阻塞主線程,就是我們平常說的卡死。

        /// <summary>
        /// 異步壓縮文件為zip壓縮包
        /// </summary>
        /// <param name="zipfile">壓縮包存儲路徑</param>
        /// <param name="filenames">文件集合</param>
        public static async void ZipFilesAsync(string zipfile, string[] filenames)
        {
            await Task.Run(() =>
            {
                ZipOutputStream s = null;
                try
                {
                    s = new ZipOutputStream(System.IO.File.Create(zipfile));

                    s.SetLevel(6); // 0 - store only to 9 - means best compression 

                    foreach (string file in filenames)
                    {
                        //打開壓縮文件 
                        FileStream fs = System.IO.File.OpenRead(file);

                        var name = Path.GetFileName(file);
                        ZipEntry entry = new ZipEntry(name);
                        entry.DateTime = DateTime.Now;
                        entry.Size = fs.Length;
                        s.PutNextEntry(entry);

                        //如果文件大于1G
                        long blockSize = 51200;

                        var size = (int)fs.Length;

                        var oneG = 1024 * 1024 * 1024;

                        if (size > oneG)
                        {
                            blockSize = oneG;
                        }
                        byte[] buffer = new byte[blockSize];

                        size = fs.Read(buffer, 0, buffer.Length);

                        s.Write(buffer, 0, size);

                        while (size < fs.Length)
                        {
                            int sizeRead = fs.Read(buffer, 0, buffer.Length);
                            s.Write(buffer, 0, sizeRead);
                            size += sizeRead;
                        }
                        s.Flush();
                        fs.Close();
                    }

                }
                catch (Exception ex)
                {
                    Console.WriteLine("異步壓縮文件出錯(cuò):" + ex.Message);
                }
                finally
                {
                    s?.Finish();
                    s?.Close();
                }
            });
        }

 

四、壓縮文件夾

    實(shí)際的應(yīng)用當(dāng)中,是文件和文件夾一起壓縮,所以這種情況,就干脆把要壓縮的東西全部放到一個(gè)文件夾,然后進(jìn)行壓縮。

 主方法如下:

/// <summary>
        /// 異步壓縮文件夾為zip壓縮包
        /// </summary>
        /// <param name="zipfile">壓縮包存儲路徑</param>
        /// <param name="sourceFolder">壓縮包存儲路徑</param>
        /// <param name="filenames">文件集合</param>
        public static async void ZipFolderAsync(string zipfile, string sourceFolder, string[] filenames)
        {
            await Task.Run(() =>
            {
                ZipOutputStream s = null;
                try
                {
                    s = new ZipOutputStream(System.IO.File.Create(zipfile));

                    s.SetLevel(6); // 0 - store only to 9 - means best compression 

                    CompressFolder(sourceFolder, s, sourceFolder);
                }
                catch (Exception ex)
                {
                    Console.WriteLine("異步壓縮文件出錯(cuò):" + ex.Message);
                }
                finally
                {
                    s?.Finish();
                    s?.Close();
                }
            });
        }

壓縮的核心方法:

   /// <summary>
          /// 壓縮文件夾
       /// </summary>
         /// <param name="source">源目錄</param>
        /// <param name="s">ZipOutputStream對象</param>
         /// <param name="parentPath">和source相同</param>
          public static void CompressFolder(string source, ZipOutputStream s, string parentPath)
          {
              string[] filenames = Directory.GetFileSystemEntries(source);
             foreach (string file in filenames)
            {
                 if (Directory.Exists(file))                 {
                     CompressFolder(file, s, parentPath);  //遞歸壓縮子文件夾
                 }
                 else
                 {
                     using (FileStream fs = System.IO.File.OpenRead(file))
                     {
                         var writeFilePath = file.Replace(parentPath, "");
                         ZipEntry entry = new ZipEntry(writeFilePath);
                         entry.DateTime = DateTime.Now;
                        entry.Size = fs.Length;
 
                        s.PutNextEntry(entry);
 
                         //如果文件大于1G
                         long blockSize = 51200;
 
                        var size = (int)fs.Length;

                         var oneG = 1024 * 1024 * 1024;

                         if (size > oneG)
                         {
                             blockSize = oneG;
                         }
                         byte[] buffer = new byte[blockSize];
 
                         size = fs.Read(buffer, 0, buffer.Length);

                       s.Write(buffer, 0, size);
 
 
                         while (size < fs.Length)
                         {
                             int sizeRead = fs.Read(buffer, 0, buffer.Length);
                            s.Write(buffer, 0, sizeRead);
                             size += sizeRead;
                         }
 
                         s.Flush();   //清除流的緩沖區(qū),使得所有緩沖數(shù)據(jù)都寫入到文件中
                         fs.Close();
                     }
                }
             }
         }

唯一需要注意的地方,可能解壓出來的目錄結(jié)構(gòu)和壓縮前的文件目錄不同,這時(shí)候檢查parentPath參數(shù),它在ZipEntry實(shí)體new的時(shí)候用,替換絕對路徑為當(dāng)前的相對路徑,也就是相對壓縮文件夾的路徑。

上面的方法比較復(fù)雜,還有一種相對簡單的方式,直接調(diào)用api:

      public static string ZipFolder(string sourceFolder, string zipFile)
        {
            string result = "";
            try
            {
                //創(chuàng)建壓縮包
                if (!Directory.Exists(sourceFolder)) return result = "壓縮文件夾不存在";

                DirectoryInfo d = new DirectoryInfo(sourceFolder);
                var files = d.GetFiles();
                if (files.Length == 0)
                {
                    //找子目錄
                    var ds = d.GetDirectories();
                    if (ds.Length > 0)
                    {
                        files = ds[0].GetFiles();
                    }
                }
                if (files.Length == 0) return result = "待壓縮文件為空";
                System.IO.Compression.ZipFile.CreateFromDirectory(sourceFolder, zipFile);
            }
            catch (Exception ex)
            {
                result += "壓縮出錯(cuò):" + ex.Message;
            }
            return result;
        }

以上就是c語言壓縮文件詳細(xì)講解的詳細(xì)內(nèi)容,更多關(guān)于c語言壓縮文件的資料請關(guān)注腳本之家其它相關(guān)文章!希望大家以后多多支持腳本之家!

相關(guān)文章

  • 探究C++中指針與數(shù)組運(yùn)算符優(yōu)先級

    探究C++中指針與數(shù)組運(yùn)算符優(yōu)先級

    C++中與指針和數(shù)組相關(guān)的運(yùn)算符優(yōu)先級,通過實(shí)際代碼示例解釋了運(yùn)算符的左結(jié)合與右結(jié)合方式,以及如何使用圓括號()來改變默認(rèn)的結(jié)合順序,文章還提供了一個(gè)優(yōu)先級表,列出了運(yùn)算符的優(yōu)先級和結(jié)合性,幫助讀者更好地理解復(fù)雜表達(dá)式中運(yùn)算符的調(diào)用順序
    2024-10-10
  • C語言代碼實(shí)現(xiàn)簡單2048游戲

    C語言代碼實(shí)現(xiàn)簡單2048游戲

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)2048游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • C++算術(shù)運(yùn)算符與類型轉(zhuǎn)換

    C++算術(shù)運(yùn)算符與類型轉(zhuǎn)換

    這篇文章主要介紹了C++算術(shù)運(yùn)算符與類型轉(zhuǎn)換,C++當(dāng)中提供5種基礎(chǔ)的算術(shù)運(yùn)算符,分別是加法、減法、乘法、除法和取模。下main我們就一起來看看下面文章得具體舉例與說明,需要的朋友可以參考一下,希望對你有所幫助
    2021-11-11
  • C++實(shí)現(xiàn)LeetCode(140.拆分詞句之二)

    C++實(shí)現(xiàn)LeetCode(140.拆分詞句之二)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(140.拆分詞句之二),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • vs2019配置Qt5開發(fā)環(huán)境(圖文教程)

    vs2019配置Qt5開發(fā)環(huán)境(圖文教程)

    本文主要介紹了如何使用visual studi2019配置qt5開發(fā)環(huán)境,以及創(chuàng)建qt項(xiàng)目,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-12-12
  • 詳解C++虛函數(shù)表存儲位置

    詳解C++虛函數(shù)表存儲位置

    相信大家知道虛表指針和虛函數(shù)存儲的位置,但對于虛函數(shù)表的存儲位置一時(shí)無法確定。本文就來和大家詳細(xì)聊聊相關(guān)內(nèi)容,希望對大家有所幫助
    2023-04-04
  • C++中析構(gòu)函數(shù)為何是虛函數(shù)

    C++中析構(gòu)函數(shù)為何是虛函數(shù)

    這篇文章主要介紹了C++中析構(gòu)函數(shù)為何是虛函數(shù)問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • C++ pair的用法案例詳解

    C++ pair的用法案例詳解

    這篇文章主要介紹了C++ pair的用法案例詳解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • C++控制臺用定時(shí)器實(shí)例代碼

    C++控制臺用定時(shí)器實(shí)例代碼

    這篇文章主要介紹了C++控制臺用定時(shí)器實(shí)例代碼,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下
    2018-02-02
  • C++虛繼承的實(shí)現(xiàn)原理由內(nèi)存布局開始講起

    C++虛繼承的實(shí)現(xiàn)原理由內(nèi)存布局開始講起

    為了解決多繼承時(shí)的命名沖突和冗余數(shù)據(jù)問題,C++提出了虛繼承,使得在派生類中只保留一份間接基類的成員,下面我們從內(nèi)存布局看看虛繼承的實(shí)現(xiàn)原理
    2022-06-06

最新評論