2

解压时出现错误

“输入不是有效的 Base-64 字符串,因为它包含非 base 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 字符串 pH = DecompressData(stream2)); 其中stream2是字符串类型..在数据库中它存储在nvarchar类型列中,同时压缩删除根标签。

第一次 xml 就像 peet 3/24/2012 Percent 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>
4

1 回答 1

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); 
   }          
}

如果我对您的理解正确,您正在将 base 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);
于 2012-07-13T17:05:10.880 回答