使用 .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 文件更改这些开关。
我正在测试从 .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 文件更改这些开关。