0

0

使用 XPath 将无序列表 HTML 标记转换为多维数组

碧海醫心

碧海醫心

发布时间:2025-08-12 17:32:01

|

904人浏览过

|

来源于php中文网

原创

使用 xpath 将无序列表 html 标记转换为多维数组

本文介绍如何使用 PHP 的 DOMDocument 和 DOMXPath 类,从 HTML 代码中提取无序列表数据,并将其转换为结构化的多维数组,最终以 JSON 格式输出。重点讲解了如何使用 XPath 查询选取特定的 HTML 元素,以及如何处理提取到的文本数据,使其符合目标数组结构。

使用 PHP 和 XPath 解析 HTML 无序列表

本教程将指导你如何使用 PHP 的 DOMDocument 和 DOMXPath 类,从 HTML 字符串中提取数据,并将其转换为多维数组。我们将以提取物流跟踪信息为例,HTML 结构包含多个

    元素,每个
      包含多个
    • 元素,每个
    • 元素包含状态、日期、地点等信息。

      1. 加载 HTML 并创建 DOMXPath 对象

      首先,我们需要将 HTML 字符串加载到 DOMDocument 对象中,并创建一个 DOMXPath 对象,以便使用 XPath 查询 HTML 元素。

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

      $html = <<
      
      
      • Status: Objeto em trânsito - por favor aguarde
      • Data : 24/10/2021 | Hora: 12:04
      • Origem: Unidade de Tratamento - Jaboatao Dos Guararapes / PE
      • Destino: Agência dos Correios - Cuitegi / PB
      • Status: Objeto em trânsito - por favor aguarde
      • Data : 19/10/2021 | Hora: 00:03
      • Origem: Unidade de Logística Integrada - Curitiba / PR
      • Destino: Unidade de Tratamento - Recife / PE
      • Status: Fiscalização aduaneira finalizada
      • Data : 18/10/2021 | Hora: 23:35
      • Local: Unidade Operacional - Curitiba / PR
      • Status: Objeto recebido pelos Correios do Brasil
      • Data : 16/10/2021 | Hora: 11:45
      • Local: Unidade de Logística Integrada - Curitiba / PR
      • Status: Objeto postado
      • Data : 14/10/2021 | Hora: 20:30
      • Local: País - /
HTML; function loadHTML_noemptywhitespace(string $html, int $extra_flags = 0, int $exclude_flags = 0): \DOMDocument { $flags = LIBXML_HTML_NODEFDTD | LIBXML_NOBLANKS | LIBXML_NONET; $flags = ($flags & ~ $exclude_flags) | $extra_flags; $domd = new \DOMDocument(); $domd->preserveWhiteSpace = false; @$domd->loadHTML('' . $html, $flags); $removeAnnoyingWhitespaceTextNodes = function (\DOMNode $node) use (&$removeAnnoyingWhitespaceTextNodes): void { if ($node->hasChildNodes()) { // Warning: it's important to do it backwards; if you do it forwards, the index for DOMNodeList might become invalidated; // that's why i don't use foreach() - don't change it (unless you know what you're doing, ofc) for ($i = $node->childNodes->length - 1; $i >= 0; --$i) { $removeAnnoyingWhitespaceTextNodes($node->childNodes->item($i)); } } if ($node->nodeType === XML_TEXT_NODE && !$node->hasChildNodes() && !$node->hasAttributes() && ! strlen(trim($node->textContent))) { //echo "Removing annoying POS"; // var_dump($node); $node->parentNode->removeChild($node); } //elseif ($node instanceof DOMText) { echo "not removed"; var_dump($node, $node->hasChildNodes(), $node->hasAttributes(), trim($node->textContent)); } }; $removeAnnoyingWhitespaceTextNodes($domd); return $domd; } $domd=loadHTML_noemptywhitespace($html); $xp=new DOMXPath($domd);

2. 使用 XPath 查询

    元素

使用 DOMXPath::query() 方法和 XPath 表达式选择所有具有 linha_status 类的

    元素。XPath 表达式 //div[contains(@class,'singlepost')]/ul 选取了 class 包含 singlepost 的 div 元素下的所有 ul 元素。
    $extracted=[];
    foreach($xp->query("//div[contains(@class,'singlepost')]/ul") as $ul){
        $ulData=[];
        foreach($xp->query("./li", $ul) as $li){
            $data = explode(":",$li->nodeValue, 2);
            $uldata[trim($data[0])] = trim($data[1]);
        }
        $extracted[]=$uldata;
    }

    3. 提取

  • 元素的数据
  • 在循环中,我们使用另一个 XPath 表达式 ./li 来选择当前

    问小白
    问小白

    免费使用DeepSeek满血版

    下载
      元素下的所有
    • 元素。然后,我们遍历每个
    • 元素,提取其文本内容。

      4. 构建多维数组

      将提取到的数据存储到 $extracted 数组中。每个

        元素的数据都存储为一个关联数组,键是
      • 标签中的冒号前的内容,值是冒号后的内容。

        5. 输出 JSON 格式

        最后,使用 json_encode() 函数将多维数组转换为 JSON 格式的字符串。为了使 JSON 字符串更易于阅读,可以使用 JSON_PRETTY_PRINT 选项。

        function json_encode_pretty($data, int $extra_flags = 0, int $exclude_flags = 0): string
        {
            // prettiest flags for: 7.3.9
            $flags = JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE | (defined("JSON_UNESCAPED_LINE_TERMINATORS") ? JSON_UNESCAPED_LINE_TERMINATORS : 0) | JSON_PRESERVE_ZERO_FRACTION | (defined("JSON_THROW_ON_ERROR") ? JSON_THROW_ON_ERROR : 0);
            $flags = ($flags | $extra_flags) & ~ $exclude_flags;
            return (json_encode($data, $flags));
        }
        echo json_encode_pretty($extracted);

        完整代码示例

        
        
        
        • Status: Objeto em trânsito - por favor aguarde
        • Data : 24/10/2021 | Hora: 12:04
        • Origem: Unidade de Tratamento - Jaboatao Dos Guararapes / PE
        • Destino: Agência dos Correios - Cuitegi / PB
        • Status: Objeto em trânsito - por favor aguarde
        • Data : 19/10/2021 | Hora: 00:03
        • Origem: Unidade de Logística Integrada - Curitiba / PR
        • Destino: Unidade de Tratamento - Recife / PE
        • Status: Fiscalização aduaneira finalizada
        • Data : 18/10/2021 | Hora: 23:35
        • Local: Unidade Operacional - Curitiba / PR
        • Status: Objeto recebido pelos Correios do Brasil
        • Data : 16/10/2021 | Hora: 11:45
        • Local: Unidade de Logística Integrada - Curitiba / PR
        • Status: Objeto postado
        • Data : 14/10/2021 | Hora: 20:30
        • Local: País - /
HTML; function json_encode_pretty($data, int $extra_flags = 0, int $exclude_flags = 0): string { // prettiest flags for: 7.3.9 $flags = JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE | (defined("JSON_UNESCAPED_LINE_TERMINATORS") ? JSON_UNESCAPED_LINE_TERMINATORS : 0) | JSON_PRESERVE_ZERO_FRACTION | (defined("JSON_THROW_ON_ERROR") ? JSON_THROW_ON_ERROR : 0); $flags = ($flags | $extra_flags) & ~ $exclude_flags; return (json_encode($data, $flags)); } function loadHTML_noemptywhitespace(string $html, int $extra_flags = 0, int $exclude_flags = 0): \DOMDocument { $flags = LIBXML_HTML_NODEFDTD | LIBXML_NOBLANKS | LIBXML_NONET; $flags = ($flags & ~ $exclude_flags) | $extra_flags; $domd = new \DOMDocument(); $domd->preserveWhiteSpace = false; @$domd->loadHTML('' . $html, $flags); $removeAnnoyingWhitespaceTextNodes = function (\DOMNode $node) use (&$removeAnnoyingWhitespaceTextNodes): void { if ($node->hasChildNodes()) { // Warning: it's important to do it backwards; if you do it forwards, the index for DOMNodeList might become invalidated; // that's why i don't use foreach() - don't change it (unless you know what you're doing, ofc) for ($i = $node->childNodes->length - 1; $i >= 0; --$i) { $removeAnnoyingWhitespaceTextNodes($node->childNodes->item($i)); } } if ($node->nodeType === XML_TEXT_NODE && !$node->hasChildNodes() && !$node->hasAttributes() && ! strlen(trim($node->textContent))) { //echo "Removing annoying POS"; // var_dump($node); $node->parentNode->removeChild($node); } //elseif ($node instanceof DOMText) { echo "not removed"; var_dump($node, $node->hasChildNodes(), $node->hasAttributes(), trim($node->textContent)); } }; $removeAnnoyingWhitespaceTextNodes($domd); return $domd; } $domd=loadHTML_noemptywhitespace($html); $xp=new DOMXPath($domd); $extracted=[]; foreach($xp->query("//div[contains(@class,'singlepost')]/ul") as $ul){ $ulData=[]; foreach($xp->query("./li", $ul) as $li){ $data = explode(":",$li->nodeValue, 2); $uldata[trim($data[0])] = trim($data[1]); } $extracted[]=$uldata; } echo json_encode_pretty($extracted); ?>

注意事项

  • 确保 HTML 结构是有效的,否则 XPath 查询可能无法正常工作。
  • @ 符号用于抑制 loadHTML 函数的警告信息,因为 HTML 可能包含无效的标记。
  • XPath 表达式可以根据实际的 HTML 结构进行调整。
  • json_encode() 函数的选项可以根据需要进行调整,例如,使用 JSON_UNESCAPED_UNICODE 选项可以避免 Unicode 字符被转义。
  • loadHTML_noemptywhitespace 函数用于移除HTML中的空白文本节点,避免解析结果中出现多余的空格。

总结

通过使用 PHP 的 DOMDocument 和 DOMXPath 类,我们可以方便地从 HTML 字符串中提取数据,并将其转换为结构化的多维数组。这种方法非常灵活,可以根据实际的 HTML 结构和数据需求进行调整。 通过本教程,你应该掌握了使用 XPath 解析 HTML 无序列表,并将其转换为 JSON 格式数据的基本方法。 这种方法可以应用于各种场景,例如,从网页中提取数据、解析 XML 文件等。

相关专题

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

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

2709

2023.09.01

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

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

1669

2023.10.11

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

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

1529

2023.10.11

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

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

974

2023.10.23

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

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

1444

2023.10.23

html怎么上传
html怎么上传

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

1235

2023.11.03

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

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

1529

2023.11.09

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

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

1307

2023.11.13

excel表格操作技巧大全 表格制作excel教程
excel表格操作技巧大全 表格制作excel教程

Excel表格操作的核心技巧在于 熟练使用快捷键、数据处理函数及视图工具,如Ctrl+C/V(复制粘贴)、Alt+=(自动求和)、条件格式、数据验证及数据透视表。掌握这些可大幅提升数据分析与办公效率,实现快速录入、查找、筛选和汇总。

0

2026.01.21

热门下载

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

相关下载

更多

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

最新文章

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

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