解压缩/压缩错误时出错

时间:2012-07-13 03:59:35

标签: c# .net gzipstream compression

我在解压缩时遇到错误

“输入不是有效的Base-64字符串,因为它包含非基本64个字符,两个以上的填充字符或填充字符中的非空白字符。”

它压缩得很好但不会减压。我看了很多其他有相同问题的例子,我觉得我跟着说的是什么,但是当我解压缩时仍然没什么。这是压缩和解压缩方法:

    public static string CompressData(string data)
    {
       byte[] bffr = Encoding.UTF8.GetBytes(data);
        var mStream = new MemoryStream();
        using (var gZipStream = new GZipStream(mStream, CompressionMode.Compress, true))
        {
            gZipStream.Write(bffr, 0, bffr.Length);
        }

        mStream.Position = 0;

        var compressedData = new byte[mStream.Length];
        mStream.Read(compressedData, 0, compressedData.Length);

        var gZipBuffer = new byte[compressedData.Length + 4];
        Buffer.BlockCopy(compressedData, 0, gZipBuffer, 4, compressedData.Length);
        Buffer.BlockCopy(BitConverter.GetBytes(bffer.Length), 0, gZipBuffer, 0, 4);
        return Convert.ToBase64String(gZipBuffer);
    }


    public static string DecompressData(string compressedData)
    {
        byte[] gZipBffr = Convert.FromBase64String(compressedData);
        using (var mStream = new MemoryStream())
        {
            int dataLength = BitConverter.ToInt32(gZipBffr , 0);
            mStream.Write(gZipBffr , 4, gZipBffr .Length - 4);

            var buffer = new byte[dataLength];

            mStream.Position = 0;
            using (var gZipStream = new GZipStream(mStream, CompressionMode.Decompress))
            {
                gZipStream.Read(buffer, 0, buffer.Length);
            }

            return Encoding.UTF8.GetString(buffer);
        }
    }

 string s = CompressData(s2.Tostring());

其中s2是XElement类型     string pH = DecompressData(stream2)); 其中stream2是类型字符串..在数据库中它存储在nvarchar类型列中 这里压缩我删除根标签。

第一次是xml                                        皮特            2012年3月24日            百分            33.3            10                           下次将另一个学生数据添加到现有的xml中,这里每次压缩时我们都必须删除父标记。

        <student>
        <data>
       <name>peet</name> 
       <date>3/24/2012</date> 
       <field>Percent</field>
       <new>33.3</new> 
       <old>10</old> 
        </data>
    <data>
       <name>raaz</name> 
       <date>3/24/2011</date> 
       <field></field>
       <new>33.3</new> 
       <old>10</old> 
    </data>
        <data>
       <name>arya</name> 
       <date>3/24/2010</date> 
       <field></field>
       <new>33.3</new> 
       <old>10</old> 
    </data>
        </student>

1 个答案:

答案 0 :(得分:2)

我认为你的DecompressData方法不太正确 - 对于小尺寸它会很好但是当数据中有多个学生数据时,我认为它会因为你的缓冲区中没有足够的空间而失败。

一个可行的方法就是这个(可能有一个比做数组副本更好的方法,你真的应该让缓冲区大于100但我把它做得很小,以确保它循环几次以确保它正常工作):

public static string DecompressData(string compressedData) 
{
    byte[] gZipBffr = Convert.FromBase64String(compressedData); 
    using (var mStream = new MemoryStream()) 
    { 
        mStream.Write(gZipBffr, 4, gZipBffr.Length - 4);      
        mStream.Position = 0;

        var bytes = new byte[0]; 
        using (var gZipStream = new GZipStream(mStream, 
                                               CompressionMode.Decompress)) 
        { 
            byte[] buffer = new byte[100];
            int read;
            while((read = gZipStream.Read(buffer, 0, buffer.Length)) > 0)
            {
                var newBytes = new byte[bytes.Length + read];
                Array.Copy(bytes, newBytes, bytes.Length);
                Array.Copy(buffer, 0, newBytes, bytes.Length, read);
                bytes = newBytes;
            }
        }
        return Encoding.UTF8.GetString(bytes); 
   }          
}

如果我已正确理解你,你将基本64位编码的压缩XML存储在数据库(nvarchar列)中,你想把它取出来,解压缩,添加一个节点,压缩它并把它放回去。

为了帮助你(可能)这里有一些代码我敲了一下就完成了这个过程(除了数据库调用 - 你可以看到你可以把它们放在哪里):

private static void UpdateStudent(XElement data, int studentId)
{
    XDocument student = ReadStudent(studentId);
    student.Root.Add(data);
    SaveStudent(studentId, student);
}

private static void SaveStudent(int studentId, XDocument data)
{
    string compressed = CompressData(data.ToString());
    SaveStudentData(studentId, compressed);
}

private static XDocument ReadStudent(int studentId)
{
    string data = GetStudentFromDatabase(studentId);
    XDocument ret;
    if (string.IsNullOrEmpty(data))
    {
        ret = XDocument.Parse("<student></student>");
    }
    else
    {
        string decompressed = DecompressData(data).TrimEnd('\0');
        ret = XDocument.Parse(decompressed);
    }
    return ret;
}

// These two methods would query the database but for demo, just store
// in a static member
private static string GetStudentFromDatabase(int studentId)
{ 
    return _data;
}

private static void SaveStudentData(int studentId, string data)
{
    _data = data;
}

private static string _data;

然后打电话给我:

UpdateStudent(XElement.Parse("<data><name>peet</name><date>3/24/2012</date><field>Percent</field><new>33.3</new><old>10</old></data>"), 1);
UpdateStudent(XElement.Parse("<data><name>raaz</name><date>3/24/2011</date><field></field><new>33.3</new><old>10</old></data>"),1);
UpdateStudent(XElement.Parse("<data><name>arya</name><date>3/24/2010</date><field></field><new>33.3</new><old>10</old></data> "), 1);
相关问题