这两个XML文件是相同的吗?

8
我目前正在使用Delphi 7编写Web服务客户端(服务本身是用C#编写的)。一切似乎都很正常。当我运行Fiddler查看客户端应用程序发送的XML时,我注意到在使用C#编写“相同”的客户端应用程序时,XML看起来不同。下面是两个XML:

一个是从Delphi 7应用程序发送的

<?xml version="1.0"?>
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/">
  <SOAP-ENV:Body SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/" xmlns:NS2="http://tempuri.org/">
    <NS1:SomeTagName xmlns:NS1="http://tempuri.org/">
      <SomeID xsi:type="xsd:int">12345</SomeID>
      <SomeStatus xsi:type="NS2:SomeStatusType">SOME_OK_STATUS</SomeStatus>
    </NS1:SomeTagName>
  </SOAP-ENV:Body>
</SOAP-ENV:Envelope>

从C#应用程序开始的一项任务。

<?xml version="1.0" encoding="utf-8"?>
<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
  <s:Body>
    <SomeTagName xmlns="http://tempuri.org/">
      <SomeID>12345</SomeID>
      <SomeStatus>SOME_OK_STATUS</SomeStatus>
    </SomeTagName>
  </s:Body>
</s:Envelope>

我对XML不是很熟悉,所以进行了一些研究,目前我能够告诉您以下几点:

  1. UTF-8是没有编码信息的文档的默认值 - 参考链接 - 这意味着在这里没有区别。
  2. XML命名空间提供了一种避免元素名称冲突的方法 - 参考链接 - 命名空间是不同的(s:和SOAP-ENV:),但已经被指定,并且据我所知也不会有任何区别。

但是模式怎么样呢?不确定。信封中有一些额外的属性或者一些ID和状态标签的数据类型。但是这些来自服务WSDL(我猜?!)。

最后的问题:

  1. 为什么用C#(vs2012)编写的应用程序不会将所有额外的模式信息添加到XML中。如果XML具有这些信息或没有这些信息,真的重要吗?
  2. 有人能告诉我这些是否可以视为相同的吗?
2个回答

1
关于你的最后第二个问题:这些XML文件不能被视为相同。原因如下:
  1. 元素 SomeIDSomeStatus 的命名空间不相同。在 Delphi XML 中,没有任何 默认命名空间 在 XML 中的任何地方指定。我们只看到以下命名空间:

    • xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    • xmlns:xsd="http://www.w3.org/2001/XMLSchema"
    • xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    • xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/"
    • xmlns:NS2="http://tempuri.org/"
    • xmlns:NS1="http://tempuri.org/"

    因此,元素 SomeIDSomeStatus 没有任何命名空间,因为它们缺少命名空间前缀。

    相反,来自 c# 应用程序的 XML 具有以下命名空间

    • xmlns:s="http://schemas.xmlsoap.org/soap/envelope/"
    • xmlns="http://tempuri.org/"

    正如您所看到的,第二个 xmlns 属性是一个默认命名空间属性,因此子元素 SomeIDSomeStatus 在 "http://tempuri.org/" 命名空间中。这意味着这些元素具有不同的 限定名称,因此它们不等效。

    (如果我必须猜测,我会认为 c# XML 是正确的,而 Delphi XML 有一个 bug。但是我无法确定,因为您提到的 SOAP 标准的 XSD 没有针对 Body 的特定模式。)

  2. Delphi XML 具有关于 SomeIDSOME_OK_STATUS 类型的附加信息。这可能对接收者有用,因为 SOAP 标准 "未强制执行这些元素的任何特定结构或解释,并且没有提供指定要执行的处理的标准手段",适用于 Body 内部的元素。但是,如果接收者已经知道这些元素中的预期内容,则这些属性可能是不必要的。

这些是我看到的XML示例之间的主要逻辑差异。
至于第一个最终问题,我们需要查看C#代码才能明确评论为什么它生成的XML看起来像那样。如果使用XmlSerializer进行序列化,它将不会为非多态字段输出xsi:type信息,除非被强制执行

0

1 - 如您所述,UTF-8 是默认值,因此可以省略。

2 - sSOAP-ENV 引用了与同一命名空间http://schemas.xmlsoap.org/soap/envelope/中指定的相同类型的对象。您可以单击链接查看对象模式。这意味着sSOAP-ENV都是Envelope类型的对象,如模式中所述。

3 - 定义SomeTagName对象类型是可选的,因为它在其他地方没有被引用,而Envelope对象则在Body声明中被引用,这使得必须定义它(通过定义名称来定义,因为它实际上是由命名空间定义的)。因此,在一个中声明了NS1,但在另一个中没有声明任何名称。在这种情况下,只需要命名空间,两者都有。

4 - 我看到的另一个区别是 Delphi 版本设置了非复杂类型(如 xsd:int)的命名空间。这是可选的,因为它在模式中设置,应该支配此操作。如果模式引用一个类型而请求/响应引用另一个类型,则会出现问题。

唯一真正使这两者不同的是,在大型请求/响应上,Delphi 版本的有效负载将更大,这可能会影响移动设备。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接