0

GZipStream用来压缩字符串,我修改了两个不同的例子来看看什么是有效的。第一个代码片段是文档中示例的大量修改版本,它只返回一个空字符串。

public static String CompressStringGzip(String uncompressed)
{
    String compressedString;
    // Convert the uncompressed source string to a stream stored in memory
    // and create the MemoryStream that will hold the compressed string
    using (MemoryStream inStream = new MemoryStream(Encoding.Unicode.GetBytes(uncompressed)),
                        outStream = new MemoryStream())
    {
        using (GZipStream compress = new GZipStream(outStream, CompressionMode.Compress))
        {
            inStream.CopyTo(compress);
            StreamReader reader = new StreamReader(outStream);
            compressedString = reader.ReadToEnd();
        }
    }
    return compressedString;

当我调试它时,我只能说什么都没有读取reader,它compressedString是空的。但是,我编写的从CodeProject 片段修改的第二种方法是成功的。

public static String CompressStringGzip3(String uncompressed)
{
    //Transform string to byte array
    String compressedString;
    byte[] uncompressedByteArray = Encoding.Unicode.GetBytes(uncompressed);

    using (MemoryStream outStream = new MemoryStream())
    {
        using (GZipStream compress = new GZipStream(outStream, CompressionMode.Compress))
        {
            compress.Write(uncompressedByteArray, 0, uncompressedByteArray.Length);
            compress.Close();
        }
        byte[] compressedByteArray = outStream.ToArray();
        StringBuilder compressedStringBuilder = new StringBuilder(compressedByteArray.Length);
        foreach (byte b in compressedByteArray)
            compressedStringBuilder.Append((char)b);
        compressedString = compressedStringBuilder.ToString();
    }
    return compressedString;
}

为什么第一个代码片段不成功,而另一个是?尽管它们略有不同,但我不知道为什么第二个代码段中的微小变化允许它工作。我正在使用的示例字符串是SELECT * FROM foods f WHERE f.name = 'chicken';

4

2 回答 2

2

我最终使用以下代码进行压缩和解压缩:

public static String Compress(String decompressed)
{
    byte[] data = Encoding.UTF8.GetBytes(decompressed);
    using (var input = new MemoryStream(data))
    using (var output = new MemoryStream())
    {
        using (var gzip = new GZipStream(output, CompressionMode.Compress, true))
        {
            input.CopyTo(gzip);
        }
        return Convert.ToBase64String(output.ToArray());
    }
}

public static String Decompress(String compressed)
{
    byte[] data = Convert.FromBase64String(compressed);
    using (MemoryStream input = new MemoryStream(data))
    using (GZipStream gzip = new GZipStream(input, CompressionMode.Decompress))
    using (MemoryStream output = new MemoryStream())
    {
        gzip.CopyTo(output);
        StringBuilder sb = new StringBuilder();
        return Encoding.UTF8.GetString(output.ToArray());

    }
}

部分问题的解释来自this question。尽管我通过将代码更改为此答案中包含的内容来解决问题,但这些行(在我的原始代码中):

foreach (byte b in compressedByteArray)
            compressedStringBuilder.Append((char)b);

是有问题的,因为正如dlev恰当地表述的那样:

您将每个字节解释为它自己的字符,而实际上并非如此。相反,您需要以下行:

string decoded = Encoding.Unicode.GetString(compressedByteArray);

基本问题是您正在转换为基于编码的字节数组,但在检索字节时忽略了该编码。

因此,问题解决了,我使用的新代码比我的原始代码简洁得多。

于 2012-06-25T19:27:43.993 回答
0

您需要将下面的代码移到第二个 using 语句之外:

using (GZipStream compress = new GZipStream(outStream, CompressionMode.Compress)) 
{ 
    inStream.CopyTo(compress); 
    outStream.Position = 0;
    StreamReader reader = new StreamReader(outStream); 
    compressedString = reader.ReadToEnd(); 
}

CopyTo() 不会将结果刷新到底层 MemoryStream。

更新

似乎 GZipStream 在处理时关闭并处理它的底层流(不是我设计类的方式)。我已经更新了上面的示例并对其进行了测试。

于 2012-06-25T15:43:31.897 回答