解压缩zip条目到另一个Zip文件(Extract zip entries to another Z

2019-09-20 22:47发布

谁能告诉我,为什么下面的代码不起作用? 我使用的邮编流的SharpZipLib API,最新版本今天DL'ed从他们的网站。 林试图使用这个逻辑合并一个压缩文件到另一个的内容,而无需在光盘上执行IO,如预期的zip文件可能包含保留文件名的窗户。 我曾与多个不同的源和目标的zip文件(那些包含保留名称和那些不)尝试这个。 代码不抛出任何异常,如果你检查之前每次写操作的缓冲区,你可以看到它包含真实的数据,但整个操作后,完成目标zip文件的大小没有改变,你可以探索它以确认没有新的文件(那些代码应该添加)实际上已经被添加到目标文件。 :(

    public static void CopyToZip(string inArchive, string outArchive)
    {

        ZipOutputStream outStream = null;
        ZipInputStream inStream = null;
        try
        {
            outStream = new ZipOutputStream(File.OpenWrite(outArchive));
            outStream.IsStreamOwner = false;
            inStream = new ZipInputStream(File.OpenRead(inArchive));
            ZipEntry currentEntry = inStream.GetNextEntry();
            while (currentEntry != null)
            {

                byte[] buffer = new byte[1024];
                ZipEntry newEntry = new ZipEntry(currentEntry.Name);
                newEntry.Size = currentEntry.Size;
                newEntry.DateTime = currentEntry.DateTime;
                outStream.PutNextEntry(newEntry);
                int size = 0;
                while ((size = inStream.Read(buffer, 0, buffer.Length)) > 0)
                {
                    outStream.Write(buffer, 0, size);
                }
                outStream.CloseEntry();

                currentEntry = inStream.GetNextEntry();
            }
            outStream.IsStreamOwner = true;
        }
        catch (Exception e)
        {
            throw e;
        }
        finally
        {
            try { outStream.Close(); }
            catch (Exception ignore) { }
            try { inStream.Close(); }
            catch (Exception ignore) { }
        }      
    }

Answer 1:

最后我用不同的API这样做。 DOTNET的拉链从http://dotnetzip.codeplex.com/ 。 下面是执行:

    public static void CopyToZip(string inArchive, string outArchive, string tempPath)
    {
        ZipFile inZip = null;
        ZipFile outZip = null;

        try
        {
            inZip = new ZipFile(inArchive);
            outZip = new ZipFile(outArchive);
            List<string> tempNames = new List<string>();
            List<string> originalNames = new List<string>();
            int I = 0;
            foreach (ZipEntry entry in inZip)
            {
                if (!entry.IsDirectory)
                {
                    string tempName = Path.Combine(tempPath, "tmp.tmp");
                    string oldName = entry.FileName;
                    byte[] buffer = new byte[4026];
                    Stream inStream = null;
                    FileStream stream = null;
                    try
                    {
                        inStream = entry.OpenReader();
                        stream = new FileStream(tempName, FileMode.Create, FileAccess.ReadWrite);
                        int size = 0;
                        while ((size = inStream.Read(buffer, 0, buffer.Length)) > 0)
                        {
                            stream.Write(buffer, 0, size);
                        }
                        inStream.Close();
                        stream.Flush();
                        stream.Close();
                        inStream = new FileStream(tempName, FileMode.Open, FileAccess.Read);

                        outZip.AddEntry(oldName, inStream);
                        outZip.Save();
                    }
                    catch (Exception exe)
                    {
                        throw exe;
                    }
                    finally
                    {
                        try { inStream.Close(); }
                        catch (Exception ignore) { }
                        try { stream.Close(); }
                        catch (Exception ignore) { }
                    }
                }
            }

        }
        catch (Exception e)
        {
            throw e;
        }
    }


Answer 2:

一个问题,我看到的是,您打开使用File.OpenWrite()输出zip文件,这将替换现有的输出文件,而不是合并新条目进去。

有给出更新使用存储器流的压缩文件的一个例子在SharpDevelop的维基一个例子。 它可以被发现在http://wiki.sharpdevelop.net/SharpZipLib_Updating.ashx#Updating_a_zip_file_in_memory_1



Answer 3:

下面是一些简单的代码,将从输入拉链读取和写入输出拉链,这潜在地已经存在。 它不需要临时数据写入到文件系统。

  public static void CopyToZip(string inArchive, string outArchive)
  {
      using (inZip = new ZipFile(inArchive),
             outZip = new ZipFile(outArchive))
      {
          Func<String,Func<String,Stream>> getInStreamReturner = (name) => {
              return new Func<String,Stream>(a){ return inZip[a].OpenReader(); };
          };
          foreach (ZipEntry entry in inZip)
          {
              if (!entry.IsDirectory)
              {
                  string zipEntryName = entry.FileName;
                  outZip.AddEntry(zipEntryName,
                                  getInStreamReturner(zipEntryName),
                                  (name, stream) => stream.Close() );
              }
          }
          outZip.Save();
      }
  }

笔记:

  1. 此方法使用ZipFile.AddEntry接受两名代表超载:一个开瓶器和更紧密。 这些函数被调用时的时间ZipFile.Save 。 前者代表需要打开并返回一个包含要拉上的数据流。 后者委托需要直接关闭流。

  2. 有必要定义getInStreamReturner Func键,以便在的时候打开右侧流ZipFile.Save 。 请记住,在zipEntryName每次循环的时间变化值。 也ZipEntry.OpenReader()打开上的实际压缩数据流,内容和-解压缩,因为它去。 你只能有一个开放的那些的,在任何一个时间,每ZipFile的。 getInStreamReturner通过环路创建一个新的功能,每次,由此产生闭合保留的值zipEntryName ,用于在所述时间基准ZipFile.Save

  3. 如果在inArchive和outArchive之间的名称冲突这种方法将失败。 为了避免这种情况你需要检查的是,不知何故避免它。 无论是图谋新的,唯一的名称,或跳过重名到outarchive添加条目。

  4. 我没有测试过这一点。


虽然这种方法不会写入到文件系统, 解压缩和重新压缩文件中的数据。 还有就是要提供一个功能DotNetZip迁移项目而没有解压缩/重新压缩跳一个开放的要求。 我还没有实现那个呢。



文章来源: Extract zip entries to another Zip file