Как получить доступ к свойствам объекта с именами, такими как целые числа?
Как получить доступ к элементу массива PHP Associative, выход которого:
[highlighting] => stdClass Object
(
[448364] => stdClass Object
(
[Data] => Array
(
[0] => Tax amount liability is .......
Я хочу получить доступ к строковому значению в ключе [0]. Я хочу сделать что-то вроде:
print myVar->highlighting->448364->Data->0
Но две цифры/целые числа там, кажется, являются проблемой.
EDIT:
Я расскажу немного истории здесь, откуда я могу получить myVar. Я использую json_decode()
что-то вроде:
$myVar = json_decode(url)
Ответы
Ответ 1
Обновлен для 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!
Пример Live.
Факт № 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 помогают выполнять рекурсивное преобразование в массив без необходимости определять какие-либо внешние функции. Однако желательно, чтобы это выглядело, у него есть недостаток "nuke" варианта № 2 и, кроме того, недостаток: если в вашем объекте есть какие-либо строки, эти строки должны быть закодированы в UTF-8 (это требование json_encode
).
Ответ 2
Просто хотел добавить к Джону красноречивое объяснение, почему это не удается. Все это потому, что при создании массива 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...
Ответ 3
Я скопировал эту функцию из сети. Если он работает так, как говорится ( "Функция преобразования объектов 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
- затем возьмите возвращаемое значение
- echo
[highlighting][448364][Data][0]
Источник: PHP stdClass для массива и массива в stdClass
Ответ 4
Последняя альтернатива полному ответу Джона:
Просто используйте json_decode(), а второй параметр - true.
$array = json_decode($url, true);
Затем он возвращает ассоциативный массив, а не объект, поэтому нет необходимости преобразовывать его после факта.
Это может быть не подходит для каждого приложения, но это действительно помогло мне легко ссылаться на свойство объекта oroginal.
Решение было найдено в этом учебнике - http://nitschinger.at/Handling-JSON-like-a-boss-in-PHP/
Привет
Ответ 5
Я боюсь, что вам не разрешено называть объекты, начинающиеся с численных значений. Переименуйте первый "448364", начинающийся с буквы.
Второй - это массив, к которому нужно обращаться с помощью скобок:
print myVar->highlighting->test_448364->Data[0]
вместо
Ответ 6
Если объект начинается с @
, например:
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']
не будет работать.