4

谁能告诉我为什么下面的代码不起作用?我正在为 Zip 流使用 SharpZipLib API,这是今天从他们的网站上下载的最新版本。我试图使用这种逻辑将一个 zip 文件的内容合并到另一个中,而不必在磁盘上执行 IO,因为预期的 zip 文件可能包含为 windows 保留的文件名。我已经尝试过使用多个不同的源和目标 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) { }
        }      
    }
4

3 回答 3

1

我最终使用不同的 API 执行此操作。来自http://dotnetzip.codeplex.com/的DotNet zip 。这是实现:

    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;
        }
    }
于 2012-08-08T05:06:14.633 回答
0

以下是一些更简单的代码,它们将从输入 zip 读取并写入可能已经存在的输出 zip。它不需要将临时数据写入文件系统。

  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. 有必要定义getInStreamReturnerFunc,以便在 时打开正确的流ZipFile.Save。请记住,zipEntryName每次循环都会更改值。还会ZipEntry.OpenReader()在实际 zi​​p 数据上打开一个流,该流会在运行时进行读取和解压缩。在任何时候,每个 ZipFile 只能打开其中一个。getInStreamReturner每次循环都会创建一个新函数,从而创建一个闭包来保留zipEntryName.ZipFile.Save

  3. 如果 inArchive 和 outArchive 之间存在名称冲突,此方法将失败。为避免这种情况,您需要检查并以某种方式避免它。要么设计一个新的、唯一的名称,要么跳过将具有重复名称的条目添加到 outarchive 中。

  4. 我没有测试过这个。


虽然这种方法不会写入文件系统,但它解压缩和重新压缩文件数据。有一个开放的请求为 DotNetZip 提供一项功能,以在没有解压缩/重新压缩跳转的情况下迁移条目。我还没有实现。

于 2012-10-30T18:06:20.723 回答
0

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

SharpDevelop Wiki 上有一个示例,提供了使用内存流更新 zip 文件的示例。它可以在http://wiki.sharpdevelop.net/SharpZipLib_Updating.ashx#Updating_a_zip_file_in_memory_1找到

于 2012-08-07T20:09:16.543 回答