在 C# 中编译包含重定义的架构

jde*_*aan 4 c# xml schema include redefinition

当读取包含<xs:redefine>标签的模式并尝试使用模式集编译它时,我收到此异常:

'SchemaLocation' must successfully resolve if <redefine>
contains any child other than <annotation>
Run Code Online (Sandbox Code Playgroud)

我尝试了许多不成功的解决方法,例如递归解析模式并将它们添加到编译之前的模式集中,甚至将它们添加为参考。架构仍然无法编译。尝试的示例(解析主 xsd,然后在调用此递归函数后尝试编译生成的“XmlSchemas”):

private void AddRecursive (XmlSchemas xsds, XmlSchema schema)
{
    foreach (XmlSchemaExternal inc in schema.Includes) {
        String schemaLocation = null;
        schemaLocation = inc.SchemaLocation;
        XmlSchema xsd;
        using (FileStream stream = new FileStream (schemaLocation, FileMode.Open, FileAccess.Read)) {
            xsd = XmlSchema.Read (stream, null);
            xsds.Add (xsd);
            xsds.AddReference (xsd);
        }
        AddRecursive (xsds, xsd);
    }
}
Run Code Online (Sandbox Code Playgroud)

处理此类模式的正确方法是什么?为什么模式编译器不能自己解析添加的模式?

Pet*_*dea 6

使用基于流的重载读取 XML 模式的问题在于 XML 模式读取器没有可用的基本 uri。如果您的 xsd:redefine 在 schemaLocation 属性中使用相对 URI,则默认解析器将无法找到正在重新定义的架构 - 因此您会收到错误消息。

我为您提供以下配置以帮助您前进,至少在了解它是如何工作的方面。将两个架构和测试脚本保存在同一文件夹中,更新脚本中的路径并运行 C# 脚本。它会给你这个输出:

QN: http://tempuri.org/XMLSchema.xsd:TRedefine, SourceUri: file:///D:/.../.../Redefine.xsd
QN: http://www.w3.org/2001/XMLSchema:anyType, SourceUri: 
Run Code Online (Sandbox Code Playgroud)

如果您更新脚本以使用基于 Stream 的重载,您将从您的帖子中收到错误消息。

Error line 20:      xset.Add(XmlSchema.Read(File.Open    (@"D:\...\...\Redefine.xsd", FileMode.Open), null));
    'SchemaLocation' must successfully resolve if <redefine> contains any child other than     <annotation>.
'SchemaLocation' must successfully resolve if <redefine> contains any child other than     <annotation>.
Error Line 21: xset.Add(XmlSchema.Read(File.Open    (@"D:\...\...\Redefine.xsd", FileMode.Open), null));
Run Code Online (Sandbox Code Playgroud)

基本架构:

<?xml version="1.0" encoding="utf-8" ?>
<xsd:schema targetNamespace="http://tempuri.org/XMLSchema.xsd"
    elementFormDefault="qualified"
    xmlns="http://tempuri.org/XMLSchema.xsd"
    xmlns:xsd="http://www.w3.org/2001/XMLSchema">
    <xsd:complexType name="TRedefine">
        <xsd:sequence>
            <xsd:element name="base" type="xsd:string"/>
        </xsd:sequence>
    </xsd:complexType>
</xsd:schema>
Run Code Online (Sandbox Code Playgroud)

重新定义的架构:

<?xml version="1.0" encoding="utf-8"?>
<!--XML Schema generated by QTAssistant/XSR Module (http://www.paschidev.com)-->
<xsd:schema xmlns="http://tempuri.org/XMLSchema.xsd" attributeFormDefault="unqualified"     elementFormDefault="qualified" targetNamespace="http://tempuri.org/XMLSchema.xsd"     xmlns:xsd="http://www.w3.org/2001/XMLSchema">
    <!-- Put a full path here.
  <xsd:redefine schemaLocation="D:\...\...\Base.xsd">
    -->
    <xsd:redefine schemaLocation="Base.xsd">
        <xsd:complexType name="TRedefine">
            <xsd:complexContent>
                <xsd:extension base="TRedefine">
                    <xsd:sequence/>
                </xsd:extension>
            </xsd:complexContent>
        </xsd:complexType>
    </xsd:redefine>
</xsd:schema>
Run Code Online (Sandbox Code Playgroud)

一个测试脚本:

using System;
using System.IO;
using System.Xml;
using System.Xml.Schema;
class Script
{
    public static void Main()
    {
        // Enter your code below
        // Generated by QTAssistant (http://www.paschidev.com)
        XmlSchemaSet xset = new XmlSchemaSet();

        // One way of doing using an XmlReader - it'll work with relative URIs.
        using(XmlReader reader = XmlReader.Create(@"D:\...\...\Redefine.xsd"))
        {
            xset.Add(XmlSchema.Read(reader, null));
        }

        // The other way, using stream, requires all external URIs - xsd:include,     xsd:import and xsd:redefine 
        // to be absolute
        //xset.Add(XmlSchema.Read(File.Open(@"D:\...\...\Redefine.xsd", FileMode.Open), null));

        xset.Compile();
        Console.WriteLine(xset.IsCompiled);
        foreach(XmlSchemaType type in xset.GlobalTypes.Values) 
        {
            Console.WriteLine("QN: {0}, SourceUri: {1}", type.QualifiedName,     type.SourceUri);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)