我正在尝试使用dotnetzip使用以下方法将XML文件存储到zip中:
private void writeHosts()
{
XmlRootAttribute root = new XmlRootAttribute(ROOTNAME_HOST);
XmlSerializer ser = new XmlSerializer(typeof(Host[]), root);
MemoryStream ms = new MemoryStream();
StreamWriter swriter = new StreamWriter(ms);
//write xml to memory stream
ser.Serialize(swriter, m_hostList.ToArray());
swriter.Flush();
//be kind, rewind (the stream)
ms.Seek(0, SeekOrigin.Begin);
//copy memory stream to zip as a file.
using (m_repo)
{
ZipEntry e = m_repo.AddEntry(FILENAME_HOST, ms);
e.IsText = true;
m_repo.Save();
}
swriter.Close();
}
然后我使用此方法重新读取XML文件:
private List<Host> readHosts()
{
XmlRootAttribute root = new XmlRootAttribute(ROOTNAME_HOST);
XmlSerializer ser = new XmlSerializer(typeof(Host[]), root);
MemoryStream ms = new MemoryStream();
StreamReader reader = new StreamReader(ms);
List<Host> retlist = new List<Host>();
//get the vuln list from the zip and read into memory
using (m_repo)
{
ZipEntry e = m_repo[FILENAME_HOST];
e.Extract(ms);
}
//rewind to the start of the stream
ms.Flush();
ms.Seek(0, SeekOrigin.Begin);
//Pull the host list from XML
Host[] ret = (Host[])ser.Deserialize(reader);
retlist.AddRange(ret);
ms.Close();
return retlist;
}
但是,此方法在e.Extract(ms)调用时抛出ZlibException - Bad状态(无效的存储块长度)。我已经阅读了足够的文档和示例,以确定这应该可行,但这也是我第一次使用dotnetzip所以......有关如何解决此问题的任何想法?
答案 0 :(得分:0)
我真的没有得到你正在做的事情,我没有看到在zip创建阶段如何创建m_repo。看起来它过于复杂了。拉链提取部分也是一样的。 您可能以奇怪的方式创建了zip,这会导致解压缩错误。我无法分辨,因为代码并非全部存在。
我认为如果你简化,你可能会避免这些问题。
假设像这样的数据传输对象类:
public class Scrilla
{
// the parts that get serialized:
[XmlElement]
public String Name { get;set; }
[XmlElement]
public DateTime FirstReport { get;set; }
[XmlElement]
public String MoreInfo { get;set; }
}
此代码将创建一个zipfile(保存到流中),该文件的XML表示形式存储在zip文件中作为条目:
var FILENAME_HOST = "self.xml";
XmlRootAttribute root = new XmlRootAttribute("root");
XmlSerializer ser = new XmlSerializer(typeof(Scrilla), root);
MemoryStream zipStream = new MemoryStream();
var scrilla = new Scrilla
{
Name = "Foop",
FirstReport = DateTime.Now,
MoreInfo = "see http://example.org"
};
// zip up:
using (var zip1 = new ZipFile())
{
zip1.AddEntry(FILENAME_HOST,(entryName,stream) => {
using (var swriter = new StreamWriter(stream))
{
ser.Serialize(swriter, scrilla);
}});
zip1.Save(zipStream);
}
// the content (an XML file) is now held in the MemoryStream
zipStream.Seek(0, SeekOrigin.Begin);
如果您想要从同一个内存流中解压缩,可以执行以下操作:
Scrilla unpacked;
// unzip:
using (var zip2 = ZipFile.Read(zipStream))
{
Stream s = zip2[FILENAME_HOST].OpenReader();
unpacked = (Scrilla) ser.Deserialize(new StreamReader(s));
}
然后证明它有效:
// prove that it worked - print out to Console
var xmlws = new System.Xml.XmlWriterSettings { OmitXmlDeclaration = true, Indent= true };
using ( var w2 = System.Xml.XmlWriter.Create(Console.Out, xmlws))
{
ser.Serialize(w2, unpacked);
}
在创建时,上面的代码使用the ZipFile.AddEntry overload that accepts a WriteDelegate。 WriteDelegate是一个在ZipFile.Save()上调用的函数,其中包含条目名称和流,您应该为该条目写入内容。有关详细信息,请查看DotNetZip文档。正如您所看到的,这个WriteDelegate只是将一个对象序列化为一个包围该流的编写器。
解压缩也需要一个快捷方式 - 它只是从可读流中读取。
在每种情况下,您都不需要创建和搜索额外的内存流。
这段代码对我有用。
如果这对您没有帮助,也许您可以提供更多详细信息。