c#whitespaces与XmlReader有关

sci*_*uff 5 c# xmlreader

我有一个简单的xml

<data>
    <node1>value1</node1>
    <node2>value2</node2>
</data>
Run Code Online (Sandbox Code Playgroud)

我正在使用IXmlSerializable来读取和写入带有DTO的xml.以下代码工作得很好

XmlReader reader;
...
while( reader.Read() ){
    Console.Write( reader.ReadElementContentAsString() );
}
// outputs value1value2
Run Code Online (Sandbox Code Playgroud)

但是,如果xml中的空格被删除,即

<data>
    <node1>value1</node1><node2>value2</node2>
</data>
Run Code Online (Sandbox Code Playgroud)

或者我使用XmlReaderSettings.IgnoreWhitespace = true;,代码只输出"value1"忽略第二个节点.当我打印解析器经过节点,我可以看到,ReadElementContentAsString移动指针EndElementnode2,但我不明白为什么要发生或如何解决它.

它是一个可能的XML解析器实现错误吗?

===============================================

这是一个示例代码和2 xml样本,可以产生不同的结果

string homedir = Path.GetDirectoryName(Application.ExecutablePath);
string xml = Path.Combine( homedir, "settings.xml" );

FileStream stream = new FileStream( xml, FileMode.Open );

XmlReaderSettings readerSettings = new XmlReaderSettings();
readerSettings.IgnoreWhitespace = false;
XmlReader reader = XmlTextReader.Create( stream, readerSettings );

while( reader.Read() ){

    if ( reader.MoveToContent() == XmlNodeType.Element && reader.Name != "data" ){

        System.Diagnostics.Trace.WriteLine(
            reader.NodeType 
            + " "
            + reader.Name
            + " " 
            + reader.ReadElementContentAsString()
        );
    }
}

stream.Close(); 
Run Code Online (Sandbox Code Playgroud)

1.)settings.xml

<?xml version="1.0"?>
<data>
    <node-1>value1</node-1>
    <node-2>value2</node-2>
</data>
Run Code Online (Sandbox Code Playgroud)

2.)settings.xml

<?xml version="1.0"?>
<data>
    <node-1>value1</node-1><node-2>value2</node-2>
</data>
Run Code Online (Sandbox Code Playgroud)

使用(1)打印

Element node-1 value1
Element node-2 value2
Run Code Online (Sandbox Code Playgroud)

使用(2)打印

Element node-1 value1
Run Code Online (Sandbox Code Playgroud)

Luc*_*uca 2

碰巧reader.Read()读取到了空白字符。忽略空格,相同的指令读取第二个元素(“gnam”XML 标记),实际上将指针带到了node2元素。

调试reader示例中调用的方法之前和之后的属性。检查NodeTypeValue属性。还检查MoveToContent方法,它非常有用。

阅读所有这些方法和属性的文档,您最终将了解XmlReader类的工作原理以及如何将其用于您的目的。是第一个谷歌结果:它包含一个非常明确的示例。

我最终得到了以下(不完整)模式:

private static void ReadXmlExt(XmlReader xmlReader, IXmlSerializableExt xmlSerializable, ReadElementDelegate readElementCallback)
{
    bool isEmpty;

    if (xmlReader == null)
        throw new ArgumentNullException("xmlReader");
    if (readElementCallback == null)
        throw new ArgumentNullException("readElementCallback");

    // Empty element?
    isEmpty = xmlReader.IsEmptyElement;
    // Decode attributes
    if ((xmlReader.HasAttributes == true) && (xmlSerializable != null))
        xmlSerializable.ReadAttributes(xmlReader);

    // Read the root start element
    xmlReader.ReadStartElement();

    // Decode elements
    if (isEmpty == false) {
        do {
            // Read document till next element
            xmlReader.MoveToContent();

            if (xmlReader.NodeType == XmlNodeType.Element) {
                string elementName = xmlReader.LocalName;

                // Empty element?
                isEmpty = xmlReader.IsEmptyElement;

                // Decode child element
                readElementCallback(xmlReader);
                xmlReader.MoveToContent();

                // Read the child end element (not empty)
                if (isEmpty == false) {
                    // Delegate check: it has to reach and end element
                    if (xmlReader.NodeType != XmlNodeType.EndElement)
                        throw new InvalidOperationException(String.Format("not reached the end element"));
                    // Delegate check: the end element shall correspond to the start element before delegate
                    if (xmlReader.LocalName != elementName)
                        throw new InvalidOperationException(String.Format("not reached the relative end element of {0}", elementName));

                    // Child end element
                    xmlReader.ReadEndElement();
                }
            } else if (xmlReader.NodeType == XmlNodeType.Text) {
                if (xmlSerializable != null) {
                    // Interface
                    xmlSerializable.ReadText(xmlReader);
                    Debug.Assert(xmlReader.NodeType != XmlNodeType.Text, "IXmlSerializableExt.ReadText shall read the text");
                } else
                    xmlReader.Skip();   // Skip text
            }
        } while (xmlReader.NodeType != XmlNodeType.EndElement);
    }
}
Run Code Online (Sandbox Code Playgroud)