为啥 SoapClient 在文档/文字样式中使用 WSDL 返回带有额外关键元素的数组?

Posted

技术标签:

【中文标题】为啥 SoapClient 在文档/文字样式中使用 WSDL 返回带有额外关键元素的数组?【英文标题】:Why does SoapClient using WSDL in document/literal style return array with extra key element?为什么 SoapClient 在文档/文字样式中使用 WSDL 返回带有额外关键元素的数组? 【发布时间】:2013-07-25 08:38:34 【问题描述】:

我们正在将 RPC/编码的 web 服务转换为 document/literal/wrapped。 WSDL(使用 nusoap)已经重写为使用新格式。

我像这样使用 php SoapClient:

new SoapClient($wsdlUrl, array(
    'cache_wsdl' => WSDL_CACHE_NONE,
    'trace' => true,
    'features' => SOAP_SINGLE_ELEMENT_ARRAYS,
));

相关的 WSDL 部分如下所示,(应该遵循 WS-I Basic Profile):

<xsd:complexType name="messages">
    <xsd:sequence>
        <xsd:element name="item" type="xsd:string" maxOccurs="unbounded"/>
    </xsd:sequence>
</xsd:complexType>

<xsd:complexType name="some_functionResponseType">
    <xsd:all>
        <xsd:element name="return" type="tns:messages" form="unqualified"/>
    </xsd:all>
</xsd:complexType>

<message name="some_functionResponse">
    <part name="parameters" element="tns:some_functionResponseType"/>
</message>

<operation name="some_function">
    <input message="tns:some_functionRequest"/>
    <output message="tns:some_functionResponse"/>
</operation>

当我提交请求时,XML 响应是这样的:

<?xml version="1.0" encoding="utf-8"?>
<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>
    <some_functionResponse xmlns="urn:toets_nl_wsdl">
      <messages xmlns="">
        <item>foo</item>
        <item>bar</item>
      </messages>
    </some_functionResponse>
  </SOAP-ENV:Body>
</SOAP-ENV:Envelope>

当我在 PHP 中转储结果对象时,它看起来像这样:

stdClass Object
(
    [messages] => stdClass Object
        (
            [item] => Array             // <-- here
                (
                    [0] => foo
                    [1] => bar
                )

        )

)

为什么结果树中有一个额外的元素“item”?当我们还在使用 RPC/encoded 时,这并不存在。

有没有办法在处理响应时删除该元素?

【问题讨论】:

您能发布 WSDL 和架构吗?当你还在使用 RPC/encoded 的时候是什么样子的? 【参考方案1】:

您的 WSDL 明确指出有一个项目的出现次数不受限制,其中包含字符串(也称为数组)。所以 PHP 只是向您呈现 WSDL 中描述的结构,并由服务器返回。

我看不出有什么问题。不要仅仅因为 Soap 的作用相同,就期望 Soap 与 RPC 相同。如果您不想要该 item 元素,请更改 WSDL 和服务 - 但这可能比将 PHP 客户端代码适合新数据结构更困难。

您甚至使用了 SOAP_SINGLE_ELEMENT_ARRAYS,这是避免检查元素是否真的是数组的好方法。

【讨论】:

WSDL 数组是根据 WS-I 规范构建的,我认为这是这里的权威来源? (ws-i.org/profiles/…)。我看到 PHP 只是从字面上简单地解析它,并且我认为在检测到这个数组结构时它可能会“更智能”。其他(可能更成熟的)SOAP 客户端(其他语言)是否以同样的方式处理这个问题? 其他语言提供了不同的变量结构,可能能够更好地覆盖这种情况,但是 PHP 只能提供数组或对象——而且 SoapClient 本身也必须将这些数据结构映射回来和转为 XML。而做“智能”优化将是错误的——这就是为什么你有 SOAP_SINGLE_ELEMENT_ARRAYS,它在不同的级别上完全禁用了这种“智能”行为。 Sven,你找到解决问题的方法了吗?【参考方案2】:

您是否完全确定它没有在 WSDL 定义中的某处指定?我确定服务器不会为了好玩而添加额外的元素:)

或者可能是文档编码风格的结果。

当返回数组类型的内容时,item 元素在 SAP Webservices(我经常使用)中非常常见,但就像我上面写的,它在 WSDL 中明确指定。

您可能还会发现,当数组只有一个元素时,item 将不是数组,因此请在您的代码中使用

if(!is_array($messages->item)) 
    $messages->item = array($messages->item);

【讨论】:

当只有一项时,PHP Soap 有一个选项 SOAP_SINGLE_ELEMENT_ARRAYS 来阻止您描述的行为。 是的,“项目”名称在 WSDL 中(也在消息中)。也许这正是您想在文档/文字中使用数组时得到的结果,但我希望我只是遗漏了一些东西。 我从未注意到 SOAP_SINGLE_ELEMENT_ARRAYS!感谢那。我一直认为是服务器决定了它:)

以上是关于为啥 SoapClient 在文档/文字样式中使用 WSDL 返回带有额外关键元素的数组?的主要内容,如果未能解决你的问题,请参考以下文章

在Word里,设置成标题的文字,更改正文后它的段落格式也跟着变了,为啥?谢谢

为啥在word中只选中一部分文字变换颜色,其它的文字都会跟着变动?

为啥word里面的图片有时候显示不出来

为啥 XSLT 文档被认为是“样式表”?

WORD2010插入题注时如果加入章节号?总是提示“ 错误!文档中没有指定样式的文字。”

flex布局 为啥文字是竖着的