100

我正在使用json_decode()类似的东西:

$myVar = json_decode($data)

这给了我这样的输出:

[highlighting] => stdClass Object
        (
            [448364] => stdClass Object
                (
                    [Data] => Array
                        (
                            [0] => Tax amount liability is ....... 

我想访问键 [0] 中的字符串值。当我尝试做类似的事情时:

print $myVar->highlighting->448364->Data->0;

我收到此错误:

解析错误:语法错误,意外的 T_DNUMBER

这两个数字/整数似乎有问题。

4

7 回答 7

305

更新为 PHP 7.2

PHP 7.2 引入了在对象和数组强制转换中转换数字键的行为更改,它修复了这种特殊的不一致性并使以下所有示例的行为都符合预期。

少一件事要混淆!


原始答案(适用于 7.2.0 之前的版本)

PHP 有一些你真的不想在里面找到自己的黑暗小巷。名称为数字的对象属性就是其中之一...

他们从未告诉过你的事

事实 #1:您不能轻易访问名称不是合法变量名的属性

$a = array('123' => '123', '123foo' => '123foo');
$o = (object)$a;
echo $o->123foo; // error

事实 #2:可以使用花括号语法访问此类属性

$a = array('123' => '123', '123foo' => '123foo');
$o = (object)$a;
echo $o->{'123foo'}; // OK!

事实 #3:但如果属性名称全为数字,则不然!

$a = array('123' => '123', '123foo' => '123foo');
$o = (object)$a;
echo $o->{'123foo'}; // OK!
echo $o->{'123'}; // error!

活生生的例子

事实 #4:好吧,除非对象一开始不是来自数组。

$a = array('123' => '123');
$o1 = (object)$a;
$o2 = new stdClass;
$o2->{'123'} = '123'; // setting property is OK

echo $o1->{'123'}; // error!
echo $o2->{'123'}; // works... WTF?

活生生的例子

很直观,你不同意吗?

你可以做什么

选项#1:手动执行

最实用的方法是将您感兴趣的对象转换回一个数组,这将允许您访问属性:

$a = array('123' => '123', '123foo' => '123foo');
$o = (object)$a;
$a = (array)$o;
echo $o->{'123'}; // error!
echo $a['123']; // OK!

不幸的是,这不能递归地工作。因此,在您的情况下,您需要执行以下操作:

$highlighting = (array)$myVar->highlighting;
$data = (array)$highlighting['448364']->Data;
$value = $data['0']; // at last!

选项#2:核选项

另一种方法是编写一个函数,将对象递归地转换为数组:

function recursive_cast_to_array($o) {
    $a = (array)$o;
    foreach ($a as &$value) {
        if (is_object($value)) {
            $value = recursive_cast_to_array($value);
        }
    }

    return $a;
}

$arr = recursive_cast_to_array($myVar);
$value = $arr['highlighting']['448364']['Data']['0'];

但是,我不相信这是一个更好的选择,因为它会不必要地转换到数组中您感兴趣的所有属性以及您感兴趣的属性。

选项#3:聪明地玩

上一个选项的替代方法是使用内置 JSON 函数:

$arr = json_decode(json_encode($myVar), true);
$value = $arr['highlighting']['448364']['Data']['0'];

JSON 函数有助于执行到数组的递归转换,而无需定义任何外部函数。不管这看起来多么理想,它具有选项 #2 的“nuke”缺点,另外还有一个缺点,即如果您的对象中有任何字符串,这些字符串必须以 UTF-8 编码(这是 的要求json_encode)。

于 2012-04-26T12:16:18.673 回答
11

只是想在乔恩雄辩的解释中添加失败的原因。这完全是因为在创建数组时,php 将键转换为整数(如果可以的话),这会导致对已转换为对象的数组进行查找问题,这仅仅是因为保留了数字键。这是有问题的,因为所有属性访问选项都期望或转换为字符串。您可以通过执行以下操作来确认这一点:

$arr = array('123' => 'abc');
$obj = (object) $arr;
$obj->{'123'} = 'abc';
print_r( $obj );

这将输出:

stdClass Object ( 
  [123] => 'abc', 
  [123] => 'abc'
)

所以该对象有两个属性键,一个是数字的(不能访问),一个是基于字符串的。这就是 Jon#Fact 4工作的原因,因为通过使用花括号设置属性意味着您始终定义基于字符串的键,而不是数字。

采用 Jon 的解决方案,但反过来,您可以通过执行以下操作从数组中生成一个始终具有基于字符串的键的对象:

$obj = json_decode(json_encode($arr));

从现在开始,您可以使用以下任一方法,因为以这种方式访问​​总是将花括号内的值转换为字符串:

$obj->{123};
$obj->{'123'};

好旧的不合逻辑的PHP...

于 2013-03-08T23:28:10.477 回答
2

对于 PHP 7

访问以数字作为属性名称的对象属性。在将数组转换为对象后最需要。

    $arr = [2,3,7];
    $o = (object) $arr;

    $t = "1";
    $t2 = 1;
    $t3 = (1);

    echo $o->{1};      // 3
    echo $o->{'1'};   // 3
    echo $o->$t;        // 3
    echo $o->$t2;       // 3
    echo $o->$t3;       // 3

    echo $o->1;       // error
    echo $o->(1);      // error
于 2019-08-03T08:32:11.527 回答
1

我从网上复制了这个功能。如果它像它说的那样工作(“将 stdClass 对象转换为多维数组的功能”),请尝试以下操作:

<?php

    function objectToArray($d) {
        if (is_object($d)) {
            // Gets the properties of the given object
            // with get_object_vars function
            $d = get_object_vars($d);
        }

        if (is_array($d)) {
            /*
            * Return array converted to object
            * Using __FUNCTION__ (Magic constant)
            * for recursive call
            */
            return array_map(__FUNCTION__, $d);
        }
        else {
            // Return array
            return $d;
        }
    }

?>
  • 首先将您的数组传递给objectToArray函数
  • 然后取返回值
  • 回声[highlighting][448364][Data][0]

来源:PHP stdClass 到 Array 和 Array 到 stdClass

于 2012-04-26T12:14:15.593 回答
1

乔恩综合答案的最后一个替代方案:

只需使用 json_decode() 并将第二个参数设置为true

$array = json_decode($url, true);

然后,这将返回一个关联数组而不是一个对象,因此无需在事后进行转换。

这可能并不适合每个应用程序,但它确实帮助我轻松地引用原始对象的属性。

在本教程中找到了解决方案 - http://nitschinger.at/Handling-JSON-like-a-boss-in-PHP/

问候

于 2017-10-27T06:52:30.890 回答
1

如果一个对象以@like开头:

SimpleXMLElement Object (
    [@attributes] => Array (
        [href] => qwertyuiop.html
        [id] => html21
        [media-type] => application/xhtml+xml
    )
)

你必须使用:

print_r($parent_object->attributes());

因为$parent_object->{'@attributes'}$parent_object['@attributes']不会工作。

于 2016-11-06T14:58:57.603 回答
0

恐怕您不允许命名以数字开头的对象。将第一个以字母开头的名称重命名为“448364”。

第二个是一个数组,它们将通过括号访问,如下所示:

print myVar->highlighting->test_448364->Data[0]

反而

于 2012-04-26T12:09:37.773 回答