PHP foreach 在 JSON 对象上无法正常工作


php foreach not working properly on json object

这个线程争论是这样做还是那样做。事实是,没有另一个,任何一个都是不够的。解决方案(以及对问题的更清晰描述(可以在这里找到:

如何在不知道它是否是数组的情况下迭代 JSON 属性?


我将以下 JSON 解析为对象(标准类(:

{
   "getdeparturesresult":{
      "departuresegment":{
         "departure":{
            "location":{
               "@id":"7461018",
               "@x":"12.523958",
               "@y":"57.938402",
               "name":"Noltorps centrum"
            },
            "datetime":"2014-12-04 23:05"
         },
         "direction":"Alingsås station",
         "segmentid":{
            "mot":{
               "@displaytype":"B",
               "@type":"BLT",
               "#text":"Buss"
            },
            "carrier":{
               "name":"Västtrafik",
               "url":"http://www.vasttrafik.se/",
               "id":"279",
               "number":"1"
            }
         }
      }
   }
}

XML中也是如此,更具可读性,无法在我的var_dump中漂亮地打印JSON...

<getdeparturesresult>
    <departuresegment>
        <departure>
            <location id="7461018" x="12.523958" y="57.938402">
                <name>Noltorps centrum</name>
            </location>
            <datetime>2014-12-04 23:05</datetime>
        </departure>
        <direction>Alingsås station</direction>
        <segmentid>
            <mot displaytype="B" type="BLT">Buss</mot>
            <carrier>
                    <name>Västtrafik</name>
                    <url>http://www.vasttrafik.se/</url>
                <id>279</id>
                <number>1</number>
            </carrier>
        </segmentid>
    </departuresegment>
</getdeparturesresult>

我通过这段代码迭代它...

        $r=array();
        $i=0;
        foreach ($apiData->getdeparturesresult->departuresegment as $m) {
            $r[$i]["depTime"] = $m->departure->datetime; //tid för avgången
            $segmentid = get_object_vars($m->segmentid->mot);
            $r[$i]["typText"] = $segmentid['#text'];
            //$r[$i]["typ"] = $segmentid['@displaytype'];
            $r[$i]["nr"] = $m->segmentid->carrier->number;
            $r[$i]["destination"] = $m->direction; //slutstation (som står på bussen, tåget etc.)
            $r[$i]["operator"] = $m->segmentid->carrier->name;
            $i++;
        }

。它运行良好,直到我得到像这样的响应,其中只有一个 DEPARTURE 元素,在这种情况下,我被抛出此错误:

(http500) Notice: Undefined property: stdClass::$departure

所以在我看来,循环尝试第二次运行,在第一行失败

$r[$i]["depTime"] = $m->departure->datetime; //tid för avgången

因为$m>离开不存在。但是,为什么$m甚至会存在呢?

-----更新-------

这是 JSON 在 DEPARTURESEGMENT 中有多个项目时的样子:

{
  "getdeparturesresult":{
     "departuresegment":[{
        "departure":{
           "location":{
              "@id":"7461018",
              "@x":"12.523958",
              "@y":"57.938402",
              "name":"Noltorps centrum"
           },
           "datetime":"2014-12-04 23:05"
        },
        "direction":"Alingsås station",
        "segmentid":{
           "mot":{
              "@displaytype":"B",
              "@type":"BLT",
              "#text":"Buss"
           },
           "carrier":{
              "name":"Västtrafik",
              "url":"http://www.vasttrafik.se/",
              "id":"279",
              "number":"1"
           }
        }
     },
     {
        "departure":{
           "location":{
              "@id":"7461018",
              "@x":"12.523958",
              "@y":"57.938402",
              "name":"Noltorps centrum"
           },
           "datetime":"2014-12-04 23:05"
        },
        "direction":"Alingsås station",
        "segmentid":{
           "mot":{
              "@displaytype":"B",
              "@type":"BLT",
              "#text":"Buss"
           },
           "carrier":{
              "name":"Västtrafik",
              "url":"http://www.vasttrafik.se/",
              "id":"279",
              "number":"1"
           }
        }
     }
     ]
  }
}

在这种情况下,我的循环工作正常。但是,当只有一个项目时,我需要像描述@chiliNUTS那样的循环,在foreach条件下更浅一级。

有没有办法编写一个循环,如果它是一个数组,它将遍历 DEPARTURESEGMENT 中的对象数组,如果不是,则只从第一个对象中提取值?我是否必须在循环之前放置一个单独的 if 子句来测试循环是否真的有必要(从而在循环中和循环外部加倍拾取我想要的值的逻辑(?

$mdeparture . 您正在循环遍历departuresegment的每个元素。 所以首先$mdeparture然后是它的direction,然后是它的segmentid。 你可能只是想要

foreach ($apiData->getdeparturesresult->departuresegment as $m)

测试

$json = '{"getdeparturesresult":{"departuresegment":{"departure":{"location":{"@id":"7461018","@x":"12.523958","@y":"57.938402","name":"Noltorps centrum"},"datetime":"2014-12-04 23:05"},"direction":"Alingsås station","segmentid":{"mot":{"@displaytype":"B","@type":"BLT","#text":"Buss"},"carrier":{"name":"Västtrafik","url":"http:'/'/www.vasttrafik.se'/","id":"279","number":"1"}}}}}';
$apiData = json_decode($json);
foreach ($apiData->getdeparturesresult->departuresegment as $key => $m) {
    echo "key:" . $key . "'r'n";
    echo '$m:' . print_r($m, true);
    echo "'r'n";
}

输出

key:departure
$m:stdClass Object
(
    [location] => stdClass Object
        (
            [@id] => 7461018
            [@x] => 12.523958
            [@y] => 57.938402
            [name] => Noltorps centrum
        )
    [datetime] => 2014-12-04 23:05
)
key:direction
$m:Alingsås station
key:segmentid
$m:stdClass Object
(
    [mot] => stdClass Object
        (
            [@displaytype] => B
            [@type] => BLT
            [#text] => Buss
        )
    [carrier] => stdClass Object
        (
            [name] => Västtrafik
            [url] => http://www.vasttrafik.se/
            [id] => 279
            [number] => 1
        )
)

只有第一个有日期时间...