使用 .NET 4.6 验证 xml 时出现 NullReferenceException

NullReferenceException when validating xml with .NET 4.6

我正在测试从 .NET 版本 4.5.1 切换到 4.6,并且 运行 在 xsd 验证中对可选属性使用唯一约束时出现 NullReferenceException。

at System.Xml.Schema.KeySequence.ToString()
at System.Xml.Schema.XmlSchemaValidator.EndElementIdentityConstraints(Object typedValue, String stringValue, XmlSchemaDatatype datatype)
at System.Xml.Schema.XmlSchemaValidator.InternalValidateEndElement(XmlSchemaInfo schemaInfo, Object typedValue)
at System.Xml.XsdValidatingReader.ProcessEndElementEvent()
at System.Xml.XsdValidatingReader.ProcessElementEvent()
at System.Xml.XsdValidatingReader.ProcessReaderEvent()
at System.Xml.XsdValidatingReader.Read()
at ConsoleApplication.Program.Main(String[] args)

这是在以 v4.5.x 为目标时运行的剥离代码,但在使用 4.6 时失败并出现 NullReferenceException。 (使用 VS2013 和 VS2015 在 Win7 上测试)。这在 xml 中合法吗?即使不是,它也应该引发一些 XmlException。

架构:

<?xml version="1.0" encoding="UTF-8"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
  <xs:element name="Enumerations">
    <xs:complexType>
      <xs:sequence>
        <xs:element name="Enum" minOccurs="0" maxOccurs="unbounded">
          <xs:complexType>
            <xs:attribute name="id" type="xs:string" use="optional"/>
          </xs:complexType>
        </xs:element>
      </xs:sequence>
    </xs:complexType>
    <xs:unique name="unique_EnumId_contraint">
      <xs:selector xpath="Enum"/>
      <xs:field xpath="@id"/>
    </xs:unique>
  </xs:element>
</xs:schema>

XML:

<?xml version="1.0" encoding="utf-8"?>
<Enumerations>
  <Enum />
  <Enum />
</Enumerations>

C#代码:

var settings = new XmlReaderSettings();
settings.ValidationType = ValidationType.Schema;
settings.Schemas.Add(null, "enumerations.xsd");

using (var xmlReader = XmlReader.Create("enumerations.xml", settings))
{
    while (xmlReader.Read())
    {
        if (xmlReader.NodeType == XmlNodeType.Element)
        {
            Console.CursorLeft = xmlReader.Depth * 4;
            Console.WriteLine(xmlReader.Name);
        }
    }
}

我可以重现这个。在我看来像是一个错误(<rant>.NET 4.6 有很多...</rant>)。你应该报告给 Microsoft Connect.

虽然这是固定的,但您可以在此处查看来源:http://referencesource.microsoft.com/#System.Xml/System/Xml/Schema/ConstraintStruct.cs,091791a9542f1952

它告诉我们可以使用 AppContext 开关克服它,因此只需在任何其他代码之前添加此代码即可:

AppContext.SetSwitch("Switch.System.Xml.IgnoreEmptyKeySequences", true);

有关此开关的更多信息,请参见此处:Mitigation: XML Schema Validation - 请注意句子:"The impact of this change should be minimal" :-)

PS:我相信您也可以使用适当的 .config 文件更改这些开关。