
本教程旨在解决php多维数组在soap请求中序列化为复杂xml时遇到的常见问题,特别是涉及嵌套结构、xml属性(如`xsi:type`)和重复元素。我们将深入探讨如何构建符合soap规范的php数组结构,并利用`spatie/array-to-xml`库高效、准确地将此数组转换为目标xml,从而避免“无法序列化结果”的错误。
在PHP开发中,将复杂的多维数组转换为SOAP协议所需的XML格式是一个常见的任务。然而,当数组包含嵌套结构、需要特定的XML属性(如xsi:type、命名空间声明)以及处理同名重复元素时,原生的DOMDocument或简单的递归函数往往难以直接满足SOAP的严格要求,容易导致诸如“unable to serialize result”的错误。
传统的递归转换函数在处理以下情况时尤其容易出错:
为了克服上述挑战,我们可以采用 spatie/array-to-xml 这样的专业库。该库提供了一种灵活的方式,通过约定特定的数组键来表示XML元素、属性、文本内容以及命名空间,从而简化了复杂XML的生成过程。
首先,通过Composer将库安装到您的项目中:
立即学习“PHP免费学习笔记(深入)”;
composer require spatie/array-to-xml
这是最关键的一步。我们需要将原始的PHP数组转换为一个特殊结构,该结构能够清晰地指示 spatie/array-to-xml 如何构建XML。该库通过以下约定来处理:
让我们根据目标SOAP XML结构来构建PHP数组:
<?php
use Spatie\ArrayToXml\ArrayToXml;
$originalData = [
"name" => "John Doe",
"date" => "2021-11-30 00:00:00.000",
"job" => "developer",
"where_from" => "france",
"address" => [
"country" => "france",
"city" => "paris",
"vat_number" => "123456" // 对应post_code
],
"items" => [
[
"cook" => "spoon", // 对应name
"clean" => "vacuum" // 对应material
]
]
];
$dataForXml = [
'SOAP-ENV:Body' => [
'ns1:Person' => [
'data' => [ // 注意:这里将原始数据包装在 'data' 节点下,符合预期XML
"name" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "John" // 假设原始数据需要拆分或调整
],
"surname" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "Doe"
],
"job" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "developer"
],
"from" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "france"
],
"address" => [
"country" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "france"
],
"city" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "paris"
],
"post_code" => [ // 对应原始数组的vat_number
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "123456"
],
'_attributes' => ['xsi:type' => 'tns:getAddress'] // address节点的属性
],
"items" => [
// 处理多个item元素,使用 __custom:KEY:INDEX 约定
'__custom:item:0' => [ // 第一个item
"name" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "spoon" // 对应原始数组的cook
],
"material" => [
'_attributes' => ['xsi:type' => 'xsd:string'],
'_value' => "vacuum" // 对应原始数组的clean
],
],
// 如果有更多item,可以继续添加 '__custom:item:1', '__custom:item:2' 等
'_attributes' => ['xsi:type' => 'tns:getItems'] // items节点的属性
],
'_attributes' => ['xsi:type' => 'tns:getPersonInfo'] // data节点的属性
],
'_attributes' => ['xmlns:ns1' => 'https://ex.pt/webservices'] // ns1:Person节点的属性
],
]
];
// 原始数据与目标数组的映射逻辑可能需要根据实际情况进行调整,
// 例如将 $originalData['name'] 拆分为 name 和 surname,
// 或者将 $originalData['address']['vat_number'] 映射到 post_code 等。
// 上述 $dataForXml 数组直接反映了期望的XML结构。在上述数组中:
使用 ArrayToXml::convert 方法将构建好的PHP数组转换为XML。在转换时,需要指定根元素的名称以及整个SOAP信封所需的全局属性。
// ... (接续上面的 $dataForXml 数组定义)
$responseXml = ArrayToXml::convert($dataForXml, [
'rootElementName' => 'SOAP-ENV:Envelope', // 指定XML的根元素
'_attributes' => [ // 为根元素添加全局属性(SOAP信封的属性)
'SOAP-ENV:encodingStyle' => 'http://schemas.xmlsoap.org/soap/encoding/',
'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:tns' => 'http://ex.pt/soap/WebServices'
],
], true, 'UTF-8'); // 第三个参数为是否格式化输出,第四个为编码
echo $responseXml;执行上述代码,将生成符合SOAP规范的XML字符串,其中包含了所有预期的元素、属性和嵌套结构。
通过 spatie/array-to-xml 库,我们可以有效地将复杂的PHP多维数组序列化为符合SOAP规范的XML。关键在于理解该库的数组结构约定,特别是如何表示XML元素、属性、文本内容以及处理命名空间和重复元素。采用这种方法,可以显著提高处理SOAP XML序列化任务的效率和准确性,避免手动DOM操作的复杂性和潜在错误。
以上就是PHP多维数组到SOAP XML序列化深度指南的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号