0

0

PHP多维数组到复杂XML结构的SOAP序列化实践

心靈之曲

心靈之曲

发布时间:2025-11-10 11:55:30

|

359人浏览过

|

来源于php中文网

原创

PHP多维数组到复杂XML结构的SOAP序列化实践

本文旨在解决php多维数组向复杂soap xml结构序列化时遇到的“无法序列化结果”问题。通过深入理解soap xml的结构要求,包括命名空间和类型属性,文章将指导您如何构建符合特定xml schema的php关联数组。我们将利用`spatie/array-to-xml`库,详细演示其安装与使用方法,确保php数据能够准确无误地转换为soap兼容的xml,从而实现可靠的web服务通信。

在现代Web服务通信中,尤其是在使用SOAP协议时,将PHP中的复杂多维数组准确地序列化为符合特定XML Schema的XML结构是一项常见的挑战。直接的递归转换方法往往难以处理SOAP XML中对命名空间、属性(如xsi:type)和复杂类型数组的严格要求,从而导致“unable to serialize result”等错误。本文将详细介绍如何通过构建符合目标XML结构的PHP数组,并结合高效的第三方库,实现PHP多维数组到SOAP XML的无缝序列化。

1. 理解SOAP XML的结构要求

SOAP(Simple Object Access Protocol)消息的XML结构通常比普通的XML更为复杂,它严格遵循WSDL(Web Services Description Language)中定义的Schema。一个典型的SOAP XML结构会包含:

  • 命名空间(Namespaces): 通过xmlns:前缀定义,用于区分不同Schema中的元素。
  • 元素类型(Element Types): 通过xsi:type属性指定,明确元素的数据类型或复杂类型。
  • 属性(Attributes): 元素的附加信息,例如SOAP-ENV:encodingStyle。
  • 复杂类型数组: 例如,SOAP-ENC:Array 类型,需要特殊的结构来表示数组中的每个项及其类型。

例如,对于一个包含人员信息和物品列表的SOAP请求,其期望的XML结构可能如下所示:


   
      
         
            John
            Doe
            
            
france paris 12345
pillow cotton

要生成这样的XML,传统的递归转换函数需要进行大量修改以支持属性和命名空间,并且对于SOAP数组类型可能力不从心。

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

2. 构建符合SOAP规范的PHP数组结构

为了精确控制生成的XML,我们需要将PHP数组组织成一种能够明确表达XML元素名称、值、属性和命名空间的特定格式。spatie/array-to-xml等库通常采用一种约定,通过特殊的键名来区分这些XML组件。

假设我们有以下原始PHP数据:

$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"
    ],
    "items" => [
        [
            "cook" => "spoon",
            "clean" => "vacuum"
        ]
    ]
];

为了将其转换为上述SOAP XML结构,我们需要将$originalData转换为以下特殊结构的PHP数组。请注意,这里的转换不仅仅是简单的键值映射,更是对数据进行重塑以匹配XML Schema。例如,originalData['items'][0]['cook']被映射为items下的第一个item的name字段。

Whimsical
Whimsical

Whimsical推出的AI思维导图工具

下载
$soapFormattedData = [
    'SOAP-ENV:Body' => [
        'ns1:Person' => [
            'info' => [ // 对应XML中的  标签,这里为了清晰,使用了 'info',实际可能需要与XML标签名一致
                "name" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "John Doe"],
                // 假设 'surname' 也是需要的字段,从原始数据中获取或补充
                "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" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "123456"], // 原始数据是vat_number
                    '_attributes' => ['xsi:type' => 'tns:getAddress'] // address 标签的属性
                ],
                "items" => [
                    // 对于数组中的每个元素,使用 '__custom:标签名:索引' 来创建同名标签
                    // 原始数据中的 "cook" 和 "clean" 被映射到 "name" 和 "material"
                    '__custom:item:1' => [
                        "name" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "spoon"],
                        "material" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "vacuum"],
                    ],
                    // 如果有多个item,可以继续添加 '__custom:item:2' 等
                    '_attributes' => ['xsi:type' => 'tns:getItems'] // items 标签的属性
                ],
                '_attributes' => ['xsi:type' => 'tns:getPersonInfo'] // data/info 标签的属性
            ],
            '_attributes' => ['xmlns:ns1' => 'https://ex.pt/webservices'] // Person 标签的属性
        ]
    ]
];

关键约定解释:

  • _attributes: 用于存储当前元素的所有XML属性,其值是一个关联数组,键为属性名,值为属性值。
  • _value: 用于存储当前元素的文本内容。
  • __custom:标签名:索引: 当需要在一个父元素下创建多个同名子元素时,可以使用此约定。例如,__custom:item:1和__custom:item:2会生成两个标签。这里的索引可以是任意唯一标识符。

3. 使用 spatie/array-to-xml 进行序列化

spatie/array-to-xml是一个功能强大且易于使用的PHP库,专门用于将PHP数组转换为XML字符串。它很好地支持了上述特殊数组结构。

3.1 安装

通过Composer安装此库:

composer require spatie/array-to-xml

3.2 序列化代码

安装完成后,您可以使用以下代码将构建好的PHP数组转换为SOAP XML:

 [
        'ns1:Person' => [
            'data' => [ // 对应XML中的  标签
                "name" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "John Doe"],
                "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" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "123456"],
                    '_attributes' => ['xsi:type' => 'tns:getAddress']
                ],
                "items" => [
                    '__custom:item:1' => [
                        "name" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "spoon"],
                        "material" => ['_attributes' => ['xsi:type' => 'xsd:string'], '_value' => "vacuum"],
                    ],
                    '_attributes' => ['xsi:type' => 'tns:getItems']
                ],
                '_attributes' => ['xsi:type' => 'tns:getPersonInfo']
            ],
            '_attributes' => ['xmlns:ns1' => 'https://ex.pt/webservices']
        ]
    ]
];

// 定义SOAP Envelope的根元素和其属性
$response = ArrayToXml::convert($soapFormattedData, [
    'rootElementName' => 'SOAP-ENV:Envelope',
    '_attributes' => [
        '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 $response;

在ArrayToXml::convert()方法中:

  • 第一个参数是待转换的PHP数组。
  • 第二个参数是一个配置数组,其中:
    • rootElementName 指定了XML的根元素名称,这里是SOAP-ENV:Envelope。
    • _attributes 定义了根元素的所有属性,包括SOAP所需的各种命名空间和编码样式。
  • 第三个参数true表示是否格式化输出,使其更具可读性。
  • 第四个参数'UTF-8'指定了XML的编码。

执行上述代码将输出符合SOAP规范的XML字符串,该字符串可以用于SOAP请求的Body部分。

4. 注意事项与最佳实践

  • 严格遵循XML Schema: 成功的SOAP序列化关键在于PHP数组结构与目标XML Schema的精确匹配。任何标签名、属性或数据类型的不匹配都可能导致序列化失败或Web服务拒绝请求。
  • 数据映射: 原始PHP数组中的数据可能需要进行转换或重命名,以适应XML Schema中定义的字段名和类型。例如,原始数组中的vat_number可能需要映射到XML中的post_code。
  • 命名空间管理: SOAP XML对命名空间有严格要求。确保所有命名空间(如xmlns:ns1、xmlns:tns等)都在适当的元素上正确定义。
  • 调试技巧: 如果遇到序列化错误,首先检查生成的XML字符串是否符合预期。可以使用在线XML验证工具或SOAP调试工具(如SoapUI)来验证XML的结构和有效性。
  • 错误处理: 在实际应用中,应包含健壮的错误处理机制,例如捕获序列化过程中可能抛出的异常。
  • 性能考量: 对于非常大的数组,序列化过程可能会消耗较多的内存和CPU。在性能敏感的场景下,可以考虑分批处理或优化数组结构。

总结

将PHP多维数组序列化为复杂的SOAP XML结构,尤其是当XML中包含命名空间、xsi:type属性和嵌套数组时,需要一种比简单递归转换更精细的方法。通过理解SOAP XML的严格要求,并采用spatie/array-to-xml库提供的特殊数组结构约定(如_attributes、_value和__custom:标签名:索引),开发者可以精确地控制生成的XML,确保其符合SOAP规范,从而实现与Web服务的成功交互。关键在于将原始PHP数据映射并重塑为与目标XML Schema完全一致的中间数组结构。

相关专题

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

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

2525

2023.09.01

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

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

1603

2023.10.11

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

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

1496

2023.10.11

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

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

952

2023.10.23

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

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

1416

2023.10.23

html怎么上传
html怎么上传

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

1234

2023.11.03

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

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

1445

2023.11.09

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

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

1306

2023.11.13

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

热门下载

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

精品课程

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

共137课时 | 8.7万人学习

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

共6课时 | 7万人学习

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

共13课时 | 0.9万人学习

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

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