0

0

PHP多维数组到SOAP XML序列化深度指南

霞舞

霞舞

发布时间:2025-11-12 10:15:18

|

950人浏览过

|

来源于php中文网

原创

PHP多维数组到SOAP XML序列化深度指南

本教程旨在解决php多维数组在soap请求中序列化为复杂xml时遇到的常见问题,特别是涉及嵌套结构、xml属性(如`xsi:type`)和重复元素。我们将深入探讨如何构建符合soap规范的php数组结构,并利用`spatie/array-to-xml`库高效、准确地将此数组转换为目标xml,从而避免“无法序列化结果”的错误。

引言:PHP数组到SOAP XML序列化的挑战

在PHP开发中,将复杂的多维数组转换为SOAP协议所需的XML格式是一个常见的任务。然而,当数组包含嵌套结构、需要特定的XML属性(如xsi:type、命名空间声明)以及处理同名重复元素时,原生的DOMDocument或简单的递归函数往往难以直接满足SOAP的严格要求,容易导致诸如“unable to serialize result”的错误。

传统的递归转换函数在处理以下情况时尤其容易出错:

  1. XML属性的映射:如何将PHP数组中的数据映射为XML元素的属性,而非子元素或文本内容。
  2. 复杂类型声明:SOAP XML通常需要通过xsi:type属性指定数据类型,这在PHP数组中很难直观表达。
  3. 同名重复元素:例如,一个items节点下有多个item节点,且每个item都有自己的子元素,传统方法可能无法正确生成。
  4. 命名空间管理:SOAP XML广泛使用命名空间,需要在各个元素上正确声明和引用。

解决方案:使用 spatie/array-to-xml 库

为了克服上述挑战,我们可以采用 spatie/array-to-xml 这样的专业库。该库提供了一种灵活的方式,通过约定特定的数组键来表示XML元素、属性、文本内容以及命名空间,从而简化了复杂XML的生成过程。

步骤一:安装 spatie/array-to-xml

首先,通过Composer将库安装到您的项目中:

立即学习PHP免费学习笔记(深入)”;

composer require spatie/array-to-xml

步骤二:构建符合XML结构的PHP数组

这是最关键的一步。我们需要将原始的PHP数组转换为一个特殊结构,该结构能够清晰地指示 spatie/array-to-xml 如何构建XML。该库通过以下约定来处理:

Closers Copy
Closers Copy

营销专用文案机器人

下载
  • _attributes 键:用于定义当前元素的XML属性。其值应为关联数组,键是属性名,值是属性值。
  • _value 键:用于定义当前元素的文本内容。
  • 命名空间前缀:直接在元素名中使用冒号(:)分隔前缀和本地名,库会自动处理命名空间声明。
  • __custom:KEY:INDEX 约定:当一个父元素下有多个同名子元素时(例如,一个items下有多个item),可以使用此约定。KEY是子元素的名称,INDEX是唯一标识符。

让我们根据目标SOAP XML结构来构建PHP数组:

 "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结构。

在上述数组中:

  • SOAP-ENV:Body、ns1:Person 等带有命名空间前缀的键名直接映射为带有命名空间的XML元素。
  • _attributes 键用于为父元素添加XML属性,例如 ns1:Person 上的 xmlns:ns1,以及 name、address、items 等元素上的 xsi:type。
  • _value 键用于指定元素的文本内容。
  • items 数组下的 __custom:item:0 是一种特殊的约定,用于生成多个名为 item 的子元素,而不会将它们合并或生成不正确的结构。这里的 0 只是一个唯一标识符,可以是任何字符串或数字。

步骤三:执行转换并获取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字符串,其中包含了所有预期的元素、属性和嵌套结构。

注意事项与最佳实践

  1. 精确映射:将原始PHP数组转换为 spatie/array-to-xml 所需的结构时,务必仔细对照目标XML Schema (XSD) 和预期的XML输出。任何元素名称、属性或嵌套层级的偏差都可能导致序列化失败或生成不合规的XML。
  2. 命名空间管理:SOAP XML对命名空间有严格要求。确保在数组中正确使用前缀(如SOAP-ENV:、ns1:、tns:),并在根元素或相关父元素的 _attributes 中声明这些命名空间(如xmlns:SOAP-ENV)。
  3. 复杂类型处理:xsi:type 属性对于SOAP消息中的复杂数据类型至关重要。确保在对应元素的 _attributes 中正确设置此属性。
  4. 重复元素:当需要生成多个同名子元素时(如 items 下的多个 item),__custom:KEY:INDEX 约定是关键。INDEX 仅用于确保数组键的唯一性,它不会出现在最终的XML中。
  5. 错误处理:在实际应用中,应将XML生成过程封装在try-catch块中,以捕获可能的转换错误。
  6. 调试:如果生成的XML不符合预期,可以逐步检查 dataForXml 数组的结构,确保它准确反映了目标XML的层次和属性。

总结

通过 spatie/array-to-xml 库,我们可以有效地将复杂的PHP多维数组序列化为符合SOAP规范的XML。关键在于理解该库的数组结构约定,特别是如何表示XML元素、属性、文本内容以及处理命名空间和重复元素。采用这种方法,可以显著提高处理SOAP XML序列化任务的效率和准确性,避免手动DOM操作的复杂性和潜在错误。

相关专题

更多
php文件怎么打开
php文件怎么打开

打开php文件步骤:1、选择文本编辑器;2、在选择的文本编辑器中,创建一个新的文件,并将其保存为.php文件;3、在创建的PHP文件中,编写PHP代码;4、要在本地计算机上运行PHP文件,需要设置一个服务器环境;5、安装服务器环境后,需要将PHP文件放入服务器目录中;6、一旦将PHP文件放入服务器目录中,就可以通过浏览器来运行它。

2015

2023.09.01

php怎么取出数组的前几个元素
php怎么取出数组的前几个元素

取出php数组的前几个元素的方法有使用array_slice()函数、使用array_splice()函数、使用循环遍历、使用array_slice()函数和array_values()函数等。本专题为大家提供php数组相关的文章、下载、课程内容,供大家免费下载体验。

1334

2023.10.11

php反序列化失败怎么办
php反序列化失败怎么办

php反序列化失败的解决办法检查序列化数据。检查类定义、检查错误日志、更新PHP版本和应用安全措施等。本专题为大家提供php反序列化相关的文章、下载、课程内容,供大家免费下载体验。

1241

2023.10.11

php怎么连接mssql数据库
php怎么连接mssql数据库

连接方法:1、通过mssql_系列函数;2、通过sqlsrv_系列函数;3、通过odbc方式连接;4、通过PDO方式;5、通过COM方式连接。想了解php怎么连接mssql数据库的详细内容,可以访问下面的文章。

948

2023.10.23

php连接mssql数据库的方法
php连接mssql数据库的方法

php连接mssql数据库的方法有使用PHP的MSSQL扩展、使用PDO等。想了解更多php连接mssql数据库相关内容,可以阅读本专题下面的文章。

1402

2023.10.23

html怎么上传
html怎么上传

html通过使用HTML表单、JavaScript和PHP上传。更多关于html的问题详细请看本专题下面的文章。php中文网欢迎大家前来学习。

1231

2023.11.03

PHP出现乱码怎么解决
PHP出现乱码怎么解决

PHP出现乱码可以通过修改PHP文件头部的字符编码设置、检查PHP文件的编码格式、检查数据库连接设置和检查HTML页面的字符编码设置来解决。更多关于php乱码的问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1440

2023.11.09

php文件怎么在手机上打开
php文件怎么在手机上打开

php文件在手机上打开需要在手机上搭建一个能够运行php的服务器环境,并将php文件上传到服务器上。再在手机上的浏览器中输入服务器的IP地址或域名,加上php文件的路径,即可打开php文件并查看其内容。更多关于php相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1303

2023.11.13

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 8.2万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 6.9万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号