1

我正在使用以下代码压缩 2 个字符串,它在压缩中效果很好。但是,当我尝试解压缩第二个字符串并且如果第一个字符串很短(大约 4 个字符)时,我得到块长度与其补码异常不匹配。这是我用于压缩的类:

    using System;
    using System.IO;
    using System.IO.Compression;
    using System.Text;

    namespace CompressString {
    internal static class StringCompressor
{
    /// <summary>
    /// Compresses the string.
    /// </summary>
    /// <param name="text">The text.</param>
    /// <returns></returns>
    public static string CompressString(string value)
    {
        //Transform string into byte[] 
        byte[] byteArray = new byte[value.Length];
        int indexBA = 0;
        foreach (char item in value.ToCharArray())
        {
            byteArray[indexBA++] = (byte)item;
        }

        //Prepare for compress
        System.IO.MemoryStream ms = new System.IO.MemoryStream();
        System.IO.Compression.GZipStream sw = new System.IO.Compression.GZipStream(ms,
        System.IO.Compression.CompressionMode.Compress);

        //Compress
        sw.Write(byteArray, 0, byteArray.Length);
        //Close, DO NOT FLUSH cause bytes will go missing...
        sw.Close();

        //Transform byte[] zip data to string
        byteArray = ms.ToArray();
        System.Text.StringBuilder sB = new System.Text.StringBuilder(byteArray.Length);
        foreach (byte item in byteArray)
        {
            sB.Append((char)item);
        }
        ms.Close();
        sw.Dispose();
        ms.Dispose();

        return sB.ToString();
    }

    /// <summary>
    /// Decompresses the string.
    /// </summary>
    /// <param name="compressedText">The compressed text.</param>
    /// <returns></returns>
    public static string DecompressString(string sData)
    {
        byte[] byteArray = new byte[sData.Length];

        int indexBa = 0;
        foreach (char item in sData)
            byteArray[indexBa++] = (byte)item;

        MemoryStream memoryStream = new MemoryStream(byteArray);
        GZipStream gZipStream = new GZipStream(memoryStream, CompressionMode.Decompress);

        byteArray = new byte[1024];

        StringBuilder stringBuilder = new StringBuilder();

        int readBytes;
        while ((readBytes = gZipStream.Read(byteArray, 0, byteArray.Length)) != 0)
        {
            for (int i = 0; i < readBytes; i++) stringBuilder.Append((char)byteArray[i]);
        } gZipStream.Close(); memoryStream.Close(); gZipStream.Dispose(); memoryStream.Dispose(); return stringBuilder.ToString();
    }
}

}

我在以下行中的 DecompressString 方法出现异常:

while ((readBytes = gZipStream.Read(byteArray, 0, byteArray.Length)) != 0)
4

3 回答 3

2

对于收到错误消息“块长度与其补码不匹配”的其他人,如果您尝试解压缩未压缩的文件,则可能会发生这种情况。

于 2015-02-23T20:53:13.887 回答
1

您的 Compress 和 Decompress 方法应该接受并返回字节数组。如果您需要生成字符串输出,您可以将字节数组转换为 base64 字符串。

例如:

// compress some string data
string stringData = GetSomeStringData();

// assumes the string data is UTF8 string - would work for ASCII encoding as well
byte[] uncompressedData = Encoding.UTF8.GetBytes(stringData);

byte[] compressedData = StringCompressor.Compress(uncompressedData);

// produce base64 encode string of compressed data
string compressedString = Convert.ToBase64String(compressedData);

// decompress base64 encoded string 
// first convert to byte array
byte[] dataToDecompress = Convert.FromBase64String(compressedString);

byte[] decompressedData = StringCompressor.Decompress(dataToDecompress);

// encode decompressed data as a UTF8 encoded string
string decompressedString = Encoding.UTF8.GetString(decompressedData);

相应地重命名和调整 Compress() 和 Decompress() 方法。如果您真的想要一个严格使用字符串的类,请在您的方法中嵌入 Base 64 编码/解码的逻辑。但是,如果您的类直接使用 'byte[]`,您的代码在其他实例中将更加灵活和可重用。

于 2013-04-25T13:39:22.553 回答
1

我得到相同的“块长度与其补码不匹配”。System.IO.InvalidDataException。

我找到了这些信息:http: //yvanrodrigues.com/content/block-length-does-not-match-its-complement

它声明:“先向前搜索 2 个字节。在压缩文件的开头有一个 2 字节签名 0x789c。”

如果您跳过这 2 个字节,它应该可以工作。

更多信息: zlib 标头是什么样的?.Net zlib 使用 .Net 4.5 膨胀

于 2016-08-30T06:48:27.260 回答