DeflateStream不解压缩数据(第一次)

时间:2010-11-11 20:27:41

标签: c# .net-4.0 deflatestream

所以这是一个奇怪的。我有这种方法来采用Base64编码的缩小字符串并返回原始数据:

public static string Base64Decompress(string base64data)
{
    byte[] b = Convert.FromBase64String(base64data);
    using (var orig = new MemoryStream(b))
    {
        using (var inflate = new MemoryStream())
        {
            using (var ds = new DeflateStream(orig, CompressionMode.Decompress))
            {
                ds.CopyTo(inflate);
                return Encoding.ASCII.GetString(inflate.ToArray());
            }
        }
    }
}

这将返回一个空字符串,除非我向ds.CopyTo(inflate)添加第二次调用。 (WTF?)

   ...
            using (var ds = new DeflateStream(orig, CompressionMode.Decompress))
            {
                ds.CopyTo(inflate);
                ds.CopyTo(inflate);
                return Encoding.ASCII.GetString(inflate.ToArray());
            }
   ...

Flush上的Close / Dispose / ds无效。)

为什么DeflateStream在第一次调用时复制0个字节?我也试过用Read()循环,但它在第一次调用时也返回零,然后在第二次调用时返回。

<小时/> 更新:这是我用来压缩数据的方法。

public static string Base64Compress(string data, Encoding enc)
{
    using (var ms = new MemoryStream())
    {
        using (var ds = new DeflateStream(ms, CompressionMode.Compress))
        {
            byte[] b = enc.GetBytes(data);
            ds.Write(b, 0, b.Length);
            ds.Flush();
            return Convert.ToBase64String(ms.ToArray());
        }
    }
}

1 个答案:

答案 0 :(得分:7)

当压缩字节不完整时(即,并非所有块都被写出),就会发生这种情况。

如果我使用带有以下Decompress方法的Base64Compress,我将收到一个带有“Unknown block type”消息的InvalidDataException。流可能已损坏。'

解压

public static string Decompress(Byte[] bytes)
{
  using (var uncompressed = new MemoryStream())
  using (var compressed = new MemoryStream(bytes))
  using (var ds = new DeflateStream(compressed, CompressionMode.Decompress))
  {
    ds.CopyTo(uncompressed);
    return Encoding.ASCII.GetString(uncompressed.ToArray());
  }
}

请注意,使用以下Compress方法时,一切都按预期工作

public Byte[] Compress(Byte[] bytes)
{
  using (var memoryStream = new MemoryStream())
  {
    using (var deflateStream = new DeflateStream(memoryStream, CompressionMode.Compress))
      deflateStream.Write(bytes, 0, bytes.Length);

    return memoryStream.ToArray();
  }
}

<强>更新

哎呀,愚蠢的我......在你处理DeflateStream之前你不能ToArray内存流(因为实际上没有实现flush(和Deflate / GZip压缩数据块);最后一个块只写在close / dispose上。

将压缩重写为:

public static string Base64Compress(string data, Encoding enc)
{
  using (var ms = new MemoryStream())
  {
    using (var ds = new DeflateStream(ms, CompressionMode.Compress))
    {
      byte[] b = enc.GetBytes(data);
      ds.Write(b, 0, b.Length);
    }

    return Convert.ToBase64String(ms.ToArray());
  }
}