使用SimpleXML和XPath迭代XML,结果顺序错误


Iterating through XML with SimpleXML and XPath, Result is in wrong order

my variables.php

$xmlFile = 'xml.xml';
$xmlFileLoad = simplexml_load_file($xmlFile);
$xmlFileLoad->registerXPathNamespace("oi", "http://www.openimmo.de");
$immobilie = $xmlFileLoad->xpath('//oi:immobilie');
$zustand_angaben_letztemodernisierung = $xmlFileLoad->xpath('/oi:openimmo/oi:immobilie/oi:zustand_angaben/oi:letztemodernisierung');
$zustand_angaben_zustand = $xmlFileLoad->xpath('/oi:openimmo/oi:immobilie/oi:zustand_angaben/oi:zustand/@zustand_art');
$zustand_angaben_alter = $xmlFileLoad->xpath('/oi:openimmo/oi:immobilie/oi:zustand_angaben/oi:alter/@alter_attr');
$zustand_angaben_erschliessung = $xmlFileLoad->xpath('/oi:openimmo/oi:immobilie/oi:zustand_angaben/oi:erschliessung/@erschl_attr');
$zustand_angaben_altlasten = $xmlFileLoad->xpath('/oi:openimmo/oi:immobilie/oi:zustand_angaben/oi:altlasten');
$zustand_angaben_baujahr = $xmlFileLoad->xpath('/oi:openimmo/oi:immobilie/oi:zustand_angaben/oi:baujahr');

XML文件(缩短版)

<?xml version="1.0" encoding="utf-8"?>
<openimmo xmlns="http://www.openimmo.de"
 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xsi:schemaLocation="http://www.openimmo.de openimmo.xsd">
    <immobilie>
        <zustand_angaben>
            <altlasten>keine</altlasten>
        </zustand_angaben>
    </immobilie>
    <immobilie>
        <zustand_angaben>
            <altlasten>keine</altlasten>
        </zustand_angaben>
    </immobilie>
    <immobilie>
        <zustand_angaben>
            <altlasten>keine</altlasten>
        </zustand_angaben>
    </immobilie>
    <immobilie>
        <zustand_angaben>
            <baujahr>ca. 1900</baujahr>
            <altlasten>nein</altlasten>
        </zustand_angaben>
    </immobilie>
    <immobilie>
        <zustand_angaben>
            <baujahr>ca. 1989</baujahr>
            <zustand zustand_art="ERSTBEZUG" />
            <alter alter_attr="ALTBAU" />
            <altlasten>keine</altlasten>
        </zustand_angaben>
    </immobilie>
</openimmo>
我show.php

<?php
    include("variables.php");
    $i = 0;
    foreach ($immobilie as $immo) {
        echo "Immobilie $i<br />";
        if(isset($zustand_angaben_zustand[$i])) {
            echo $zustand_angaben_zustand[$i] . "</br>";
        }
        if(isset($zustand_angaben_alter[$i])) {
            echo $zustand_angaben_alter[$i] . "</br>";
        }
        if(isset($zustand_angaben_erschliessung[$i])) {
            echo $zustand_angaben_erschliessung[$i] . "</br>";
        }
        if(isset($zustand_angaben_altlasten[$i])) {
            echo $zustand_angaben_altlasten[$i] . "</br>";
        }
        if(isset($zustand_angaben_baujahr[$i])) {
            echo $zustand_angaben_baujahr[$i] . "</br>";
        }   
        echo "<br /><br />";
        $i++;
    }
?>

当我执行代码时,结果将是下面的

Immobilie 0
ERSTBEZUG
ALTBAU
keine
5
A
5
A++
ca. 1900
Immobilie 1
keine
ca. 1989
Immobilie 2
keine
Immobilie 3
nein
Immobilie 4
keine

但是我希望它按照xml中的顺序排列,所以不动0只有keine,不动1和2也是,第三个1900和nein最后但并非最不重要的不动4具有xml中的所有属性…

我做错了什么?

使用相对于当前immobilie元素的XPath表达式:

$i = 0;
foreach ($immobilie as $immo) {
    $immo->registerXPathNamespace("oi", "http://www.openimmo.de");
    $zustand_angaben_letztemodernisierung = $immo->xpath('./oi:zustand_angaben/oi:letztemodernisierung');
    $zustand_angaben_zustand = $immo->xpath('./oi:zustand_angaben/oi:zustand/@zustand_art');
    $zustand_angaben_alter = $immo->xpath('./oi:zustand_angaben/oi:alter/@alter_attr');
    $zustand_angaben_erschliessung = $immo->xpath('./oi:zustand_angaben/oi:erschliessung/@erschl_attr');
    $zustand_angaben_altlasten = $immo->xpath('./oi:zustand_angaben/oi:altlasten');
    $zustand_angaben_baujahr = $immo->xpath('./oi:zustand_angaben/oi:baujahr');
    echo "Immobilie $i<br />";
    if(isset($zustand_angaben_zustand[0])) {
        echo $zustand_angaben_zustand[0] . "</br>";
    }
    if(isset($zustand_angaben_alter[0])) {
        echo $zustand_angaben_alter[0] . "</br>";
    }
    if(isset($zustand_angaben_erschliessung[0])) {
        echo $zustand_angaben_erschliessung[0] . "</br>";
    }
    if(isset($zustand_angaben_altlasten[0])) {
        echo $zustand_angaben_altlasten[0] . "</br>";
    }
    if(isset($zustand_angaben_baujahr[0])) {
        echo $zustand_angaben_baujahr[0] . "</br>";
    }   
    echo "<br /><br />";
    $i++;
}

eval.in demo

当并行运行xpath查询时,它们的顺序彼此相邻。

对于您的特定示例,您还没有真正需要xpath,该文档看起来是SimpleXMLElement的完美工作,您可以直接从中检索数据。

如果您遍历您感兴趣的对象(这里最有可能的是each immobile 元素),然后提取数据(过滤掉未设置的元素),它可以类似于以下示例:

$xml = simplexml_load_string($string);
foreach ($xml->immobilie as $immobilie) {
    $angaben = $immobilie->zustand_angaben;
    $data = [
        'letzte_modernisierung' => $angaben->letztemodernisierung,
        'zustand_art'           => $angaben->zustand['zustand_art'],
        'alter'                 => $angaben->alter['alter_attr'],
        'erschliessung'         => $angaben->erschliessung['erschl_attr'],
        'altlasten'             => $angaben->altlasten,
        'baujahr'               => $angaben->baujahr,
    ];
    echo " * Immobilie:'n";
    foreach(array_filter($data) as $label => $wert) {
        echo "   - $label: $wert'n";
    }
}

输出为(也是:online-demo):

 * Immobilie:
   - altlasten: keine
 * Immobilie:
   - altlasten: keine
 * Immobilie:
   - altlasten: keine
 * Immobilie:
   - altlasten: nein
   - baujahr: ca. 1900
 * Immobilie:
   - zustand_art: ERSTBEZUG
   - alter: ALTBAU
   - altlasten: keine
   - baujahr: ca. 1989

Xpath一点也不坏,对于像您在这里看到的场景,我建议您使用DOMDocument而不是SimpleXMLElement,因为使用Xpath更灵活,这要归功于evaluate方法DOMXpath

在任何情况下,您都可以进一步阅读这个问题,它显示了一个DOMDocument和一个SimpleXMLElement基于Xpath的XML文档数据检索的解决方案:

  • PHP XPath。将复杂XML转换为数组