提取文本/ XML文件时出现Dotnetzip错误状态错误

时间:2011-05-24 21:01:48

标签: c# dotnetzip

我正在尝试使用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所以......有关如何解决此问题的任何想法?

1 个答案:

答案 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只是将一个对象序列化为一个包围该流的编写器。

解压缩也需要一个快捷方式 - 它只是从可读流中读取。

在每种情况下,您都不需要创建和搜索额外的内存流。

这段代码对我有用。

如果这对您没有帮助,也许您可​​以提供更多详细信息。