【问题标题】:Convert unordered list html markup into a multidimensional array using xpath使用 xpath 将无序列表 html 标记转换为多维数组
【发布时间】:2021-12-11 17:40:05
【问题描述】:

我正在尝试创建一个数组来重现以下代码:

<div class="singlepost">
    
<ul class="linha_status" style="">
<li>Status: <b>Objeto em trânsito - por favor aguarde</b></li>
<li>Data  : 24/10/2021 | Hora: 12:04</li>           
<li>Origem: Unidade de Tratamento - Jaboatao Dos Guararapes / PE</li>
<li>Destino: Agência dos Correios - Cuitegi / PB</li>
</ul>

<ul class="linha_status" style="">
<li>Status: <b>Objeto em trânsito - por favor aguarde</b></li>
<li>Data  : 19/10/2021 | Hora: 00:03</li>           
<li>Origem: Unidade de Logística Integrada - Curitiba / PR</li>
<li>Destino: Unidade de Tratamento - Recife / PE</li>
</ul>

<ul class="linha_status" style="">
<li>Status: <b>Fiscalização aduaneira finalizada</b></li>
<li>Data  : 18/10/2021 | Hora: 23:35</li>
<li>Local: Unidade Operacional - Curitiba / PR</li>
</ul>

<ul class="linha_status" style="">
<li>Status: <b>Objeto recebido pelos Correios do Brasil</b></li>
<li>Data  : 16/10/2021 | Hora: 11:45</li>
<li>Local: Unidade de Logística Integrada - Curitiba / PR</li>
</ul>

<ul class="linha_status" style="">
<li>Status: <b>Objeto postado</b></li>
<li>Data  : 14/10/2021 | Hora: 20:30</li>
<li>Local: País -  / </li>
</ul>

</div>

我正在使用 xpath 和 foreach 来创建数组,但结果并不好......它正在工作,但不是我需要的输出,这是我编写的代码:

$doc = new DOMDocument();
$doc->loadHTML($htmlString);
$xpath = new DOMXPath($doc);

$geral = $xpath->evaluate('//ul[@class="linha_status"]');

foreach ($geral as $name) {
    $total[] = $name->nodeValue;
}
var_dump($total);

我的实际代码会产生这个输出:

  array(5) {
    [0] => string(195)
    " 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" 
    [1] => string(189)
    " 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" 
    [2] => string(128)
    " Status: Fiscalização aduaneira finalizada Data : 18/10/2021 | Hora: 23:35 Local: Unidade Operacional - Curitiba / PR" 
    [3] => string(145)
    " Status: Objeto recebido pelos Correios do Brasil Data : 16/10/2021 | Hora: 11:45 Local: Unidade de Logística Integrada - Curitiba / PR" 
    [4] => string(83)
    " Status: Objeto postado Data : 14/10/2021 | Hora: 20:30 Local: País - / "
  }

这是我想要的输出:

"eventos": [{
    "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 - /"
  }]

【问题讨论】:

    标签: php arrays xpath foreach html-parsing


    【解决方案1】:
    $total = [];
    $ind = 0;
    foreach ($geral as $name) {
        $s = explode("\n",$name->nodeValue);
        foreach($s as  $ss){
            if(str_contains($ss,"Status: ")){
                $total[$ind]["status"] = str_replace('Status: ','',$ss);
            }
            if(str_contains($ss,"Data  : ")){
                
                $data = str_replace('Data  : ','',$ss);
                $data = str_replace('Hora: ','',$data);
                $data = explode(" | ",$data);
                $total[$ind]["data"] = $data[0];
                $total[$ind]["hora"] = $data[1];
            }
            if(str_contains($ss,"Origem: ")){
                $total[$ind]["origem"] = str_replace('Origem: ','',$ss);
            }
            if(str_contains($ss,"Destino: ")){
                $total[$ind]["destino"] = str_replace('Destino: ','',$ss);
            }
            if(str_contains($ss,"Local: ")){
                $total[$ind]["local"] = str_replace('Local: ','',$ss);
            }
        }
        $ind++;
    }
    
    print_r($total);
    

    只要确保在 li 之后有一个新行。 HTML 上的不一致可能会破坏输出。很抱歉。

    PHP v8.0

    【讨论】:

    • 您好 Percian,感谢您的回复,但它不起作用,我收到错误 500(我使用的是 PHP 7.3)
    • 我觉得你应该explode(":",$nodeValue, 2)
    • str_contains 仅适用于 8.0 版。可以在以后的php版本中使用str_pos。
    • 对.. 我做了更改(将所有“str_contains”替换为“str_pos”)但一直不工作=/知道吗?
    • @hanshenrik 嘿,谢谢你的评论,但你能解释一下我需要把这段代码放在哪里吗?
    【解决方案2】:

    也许

    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('<?xml encoding="UTF-8">' . $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);
    

    哪个打印:

    [
        {
            "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",
            "Origem": "Unidade de Logística Integrada - Curitiba / PR",
            "Destino": "Unidade de Tratamento - Recife / PE",
            "Local": "Unidade Operacional - Curitiba / PR"
        },
        {
            "Status": "Objeto recebido pelos Correios do Brasil",
            "Data": "16/10/2021 | Hora: 11:45",
            "Origem": "Unidade de Logística Integrada - Curitiba / PR",
            "Destino": "Unidade de Tratamento - Recife / PE",
            "Local": "Unidade de Logística Integrada - Curitiba / PR"
        },
        {
            "Status": "Objeto postado",
            "Data": "14/10/2021 | Hora: 20:30",
            "Origem": "Unidade de Logística Integrada - Curitiba / PR",
            "Destino": "Unidade de Tratamento - Recife / PE",
            "Local": "País -  /"
        }
    ]
    

    【讨论】:

    • 嘿@hanshenrik,只有一件事,我试图理解你的代码来分隔这些值:“数据”:“14/10/2021 | Hora:20:30”到>“数据” : "14/10/2021" 和 "Hora: 20:30", 请再给我一个帮助好吗?我真的需要将这些值分开来插入 BD ;/
    • 检查@Percian 如何解决该特定问题。类似$data = explode(":",$li-&gt;nodeValue, 2); if(trim($data[0]) === "Data"){ $data[1]=explode(" | Hora: ",$data[1]);$uldata["Data"]=$data[1][0];$uldata["Hora"]=$data[1][1]; }else{ $uldata[trim($data[0])] = trim($data[1]); }
    【解决方案3】:

    如果使用两次 xpath,解决方案会容易一些。一次用于 ul 标签,一次用于底层 li 标签。拆分只需用explode 完成。

    $doc = new DOMDocument();
    $doc->loadHTML($htmlString);
    $xpath = new DOMXPath($doc);
    
    $geral = $xpath->query('//ul[@class="linha_status"]');
    
    $total = [];
    foreach ($geral as $node) {
      $sArr = [];
      $li = $xpath->query('li',$node);
      foreach($li as $item){
        $liVal = $item->nodeValue;
        $parts = explode("|",$liVal);
        foreach($parts as $part){
          list($key,$val) = explode(':',$part);
          $sArr[trim($key)] = trim($val);
        }
      }
      $total[] = $sArr;
    }
    
    $result = json_encode($total, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
    

    sandboxhttps://3v4l.org/3E2G3 中亲自尝试。

    【讨论】:

      【解决方案4】:

      为了方便起见,我绝对支持 @jspit 使用嵌套 xpath 调用的建议,但是我更喜欢一些不同的编码选择。以下是我的 sn-p 的细分:

      1. 使用 UTF-8 编码加载文档以保留多字节字符
      2. 使用 xpath 迭代所有具有限定类的 &lt;ul&gt; 标记
      3. 使用 xpath 迭代所有嵌套在合格 &lt;ul&gt; 中的 !&lt;li&gt; 标签
      4. 用竖线分割&lt;li&gt; 文本以形成 1 个或多个段 - 不需要限制器
      5. 用第一个出现的冒号分割每个段 - 将爆炸限制为 2 部分至关重要,因为有些段包含多个冒号;在此爆炸过程中删除空格无需稍后再调用trim() 两次
      6. 将键值对推入结果数组,其中包含与父 ul 相关的一级索引。

      代码:(Demo)

      $result = [];
      $doc = new DOMDocument;
      $doc->loadHTML('<?xml encoding="UTF-8">' . $htmlString);
      $xpath = new DOMXPath($doc);
      foreach ($xpath->query('//ul[@class="linha_status"]') as $i => $ul) {
          foreach ($xpath->query('li', $ul) as $li) {
              foreach (explode("|", $li->nodeValue) as $segment) {
                  [$key, $result[$i][$key]] = preg_split('/\s*:\s*/', trim($segment), 2);
              }
          }
      }
      var_export($result);
      

      输出:

      array (
        0 => 
        array (
          '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',
        ),
        1 => 
        array (
          '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',
        ),
        2 => 
        array (
          'Status' => 'Fiscalização aduaneira finalizada',
          'Data' => '18/10/2021',
          'Hora' => '23:35',
          'Local' => 'Unidade Operacional - Curitiba / PR',
        ),
        3 => 
        array (
          'Status' => 'Objeto recebido pelos Correios do Brasil',
          'Data' => '16/10/2021',
          'Hora' => '11:45',
          'Local' => 'Unidade de Logística Integrada - Curitiba / PR',
        ),
        4 => 
        array (
          'Status' => 'Objeto postado',
          'Data' => '14/10/2021',
          'Hora' => '20:30',
          'Local' => 'País -  /',
        ),
      )
      

      【讨论】:

        猜你喜欢
        • 2014-06-28
        • 2020-02-06
        • 1970-01-01
        • 2020-02-28
        • 1970-01-01
        • 2013-05-17
        • 1970-01-01
        • 2015-08-18
        • 2013-08-09
        相关资源
        最近更新 更多