4

次のコードが機能しない理由を誰か教えてもらえますか? 私は Zip ストリームに SharpZipLib API を使用しています。最新バージョンは、今日サイトから DL されています。意図したzipファイルにはWindows用に予約されたファイル名が含まれている可能性があるため、このロジックを使用して、ディスクでIOを実行することなく、あるzipファイルの内容を別の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) { }
        }      
    }
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、オープナーとクローザーの 2 つのデリゲートを受け入れるオーバーロードを使用します。これらの関数は の時点で呼び出されZipFile.Saveます。前のデリゲートは、圧縮するデータを含むストリームを開いて返す必要があります。後者のデリゲートは、ストリームを閉じるだけで済みます。

  2. getInStreamReturnerの時点で正しいストリームを開くには、 Funcを定義する必要がありますZipFile.SavezipEntryNameループを通過するたびに値が変更されることに注意してください。またZipEntry.OpenReader()、実際の zip データのストリームを開きます。これは、進行中に読み取りと解凍を行います。ZipFile ごとに、一度に 1 つのみ開くことができます。getInStreamReturnerは、ループを通過するたびに新しい関数を作成します。これにより、zipEntryNameの時点での参照用の値を保持するクロージャが作成されZipFile.Saveます。

  3. inArchive と outArchive の間に名前の衝突がある場合、このアプローチは失敗します。それを避けるには、それを確認して、どうにかして回避する必要があります。新しい一意の名前を作成するか、重複する名前のエントリをアウトアーカイブに追加するのをスキップしてください。

  4. 私はこれをテストしていません。


このアプローチはファイルシステムに書き込みませんが、ファイルデータの解凍と再圧縮を行います。その解凍/再圧縮ジャンプなしでエントリを移行する機能を DotNetZip に提供するオープン リクエストがあります。私はまだそれを実装していません。

于 2012-10-30T18:06:20.723 に答える