我在解压缩时遇到错误
“输入不是有效的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>
答案 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);