Итак, я бродил по php.net в поисках информации о сериализации объектов PHP в JSON, когда наткнулся на новый интерфейс JsonSerializable . Это только PHP> = 5.4 , и я работаю в среде 5.3.x.
Как достигается такая функциональность в PHP <5.4 ?
Я еще не очень много работал с JSON, но я пытаюсь поддерживать уровень API в приложении, и сброс объекта данных ( который в противном случае был бы отправлен в представление ) в JSON был бы идеальным.
Если я попытаюсь сериализовать объект напрямую, он вернет пустую строку JSON; это потому, что я предполагаю, json_encode()
что не знает, что, черт возьми, делать с объектом. Должен ли я рекурсивно уменьшить объект в массив, а затем закодировать что ?
пример
$data = new Mf_Data();
$data->foo->bar['hello'] = 'world';
echo json_encode($data)
создает пустой объект:
{}
var_dump($data)
однако работает как ожидалось:
object(Mf_Data)#1 (5) {
["_values":"Mf_Data":private]=>
array(0) {
}
["_children":"Mf_Data":private]=>
array(1) {
[0]=>
array(1) {
["foo"]=>
object(Mf_Data)#2 (5) {
["_values":"Mf_Data":private]=>
array(0) {
}
["_children":"Mf_Data":private]=>
array(1) {
[0]=>
array(1) {
["bar"]=>
object(Mf_Data)#3 (5) {
["_values":"Mf_Data":private]=>
array(1) {
[0]=>
array(1) {
["hello"]=>
string(5) "world"
}
}
["_children":"Mf_Data":private]=>
array(0) {
}
["_parent":"Mf_Data":private]=>
*RECURSION*
["_key":"Mf_Data":private]=>
string(3) "bar"
["_index":"Mf_Data":private]=>
int(0)
}
}
}
["_parent":"Mf_Data":private]=>
*RECURSION*
["_key":"Mf_Data":private]=>
string(3) "foo"
["_index":"Mf_Data":private]=>
int(0)
}
}
}
["_parent":"Mf_Data":private]=>
NULL
["_key":"Mf_Data":private]=>
NULL
["_index":"Mf_Data":private]=>
int(0)
}
Дополнение
1)
Итак, это toArray()
функция, которую я разработал для Mf_Data
класса:
public function toArray()
{
$array = (array) $this;
array_walk_recursive($array, function (&$property) {
if ($property instanceof Mf_Data) {
$property = $property->toArray();
}
});
return $array;
}
Однако, поскольку Mf_Data
объекты также имеют ссылку на их родительский ( содержащий ) объект, это не удается с рекурсией. Работает как шарм, когда я удаляю _parent
ссылку.
2)
Чтобы продолжить, последняя функция для преобразования сложного объекта дерева-узла, с которым я работал, была:
// class name - Mf_Data
// exlcuded properties - $_parent, $_index
public function toArray()
{
$array = get_object_vars($this);
unset($array['_parent'], $array['_index']);
array_walk_recursive($array, function (&$property) {
if (is_object($property) && method_exists($property, 'toArray')) {
$property = $property->toArray();
}
});
return $array;
}
3)
Я снова продолжаю работу, немного чище реализации. Использование интерфейсов для instanceof
проверки кажется намного чище method_exists()
( однако method_exists()
перекрестное наследование / реализация ).
Использование unset()
тоже показалось немного запутанным, и кажется, что логику следует реорганизовать в другой метод. Тем не менее, эта реализация делает копирование массива свойства ( из - заarray_diff_key
), так что - то рассмотреть.
interface ToMapInterface
{
function toMap();
function getToMapProperties();
}
class Node implements ToMapInterface
{
private $index;
private $parent;
private $values = array();
public function toMap()
{
$array = $this->getToMapProperties();
array_walk_recursive($array, function (&$value) {
if ($value instanceof ToMapInterface) {
$value = $value->toMap();
}
});
return $array;
}
public function getToMapProperties()
{
return array_diff_key(get_object_vars($this), array_flip(array(
'index', 'parent'
)));
}
}
источник
JsonSerializable
Ответы:
изменить : в настоящее время 2016-09-24, а PHP 5.4 был выпущен 2012-03-01, и поддержка закончилась 2015-09-01. Тем не менее, этот ответ, похоже, набирает голоса. Если вы все еще используете PHP <5.4, вы создаете угрозу безопасности и ставите под угрозу свой проект . Если у вас нет веских причин оставаться на <5.4 или даже уже использовать версию> = 5.4, не используйте этот ответ , а просто используйте PHP> = 5.4 (или, вы знаете, недавний) и реализуйте интерфейс JsonSerializable.
Вы должны определить функцию, например, с именем
getJsonData();
, которая будет возвращать либо массив,stdClass
объект или какой-либо другой объект с видимыми параметрами, а не частными / защищенными, и выполнитеjson_encode($data->getJsonData());
. По сути, реализовать функцию из 5.4, но вызывать ее вручную.Что-то вроде этого будет работать, как
get_object_vars()
вызывается изнутри класса, имея доступ к закрытым / защищенным переменным:источник
stdClass
? Я думаю в направлении Reflection , но если нет, я просто придумаю что-нибудь, чтобы рекурсивно это выполнить.getJsonData()
функции, вы можете просто вызватьget_object_vars()
и просмотреть этот результат в поисках дополнительных объектов._parent
свойство, поэтому по дереву можно пройти до корня. Смотрите мое редактирование для обновления; возможно, мне следует задать другой вопрос, поскольку эта проблема теперь абстрагирована от моего оригинала.unset($array['_parent']);
перед прогулкой должно помочь.$parent
как пользовательские данныеarray_walk_recursive()
. Просто красиво! Кроме того, это$array["\0class\0property"]
из-за загрязнения нулевым байтом, потому что я использовал кастинг. Думаю перейду наget_object_vars()
.В простейших случаях должна работать подсказка типов:
источник
json_encode()
будет кодировать только общедоступные переменные-члены. поэтому, если вы хотите включить частное, когда вам нужно сделать это самостоятельно (как предлагали другие)источник
Следующий код выполняет свою работу с использованием отражения. Предполагается, что у вас есть геттеры для свойств, которые вы хотите сериализовать.
источник
Просто реализуйте интерфейс, предоставленный PHP JsonSerializable .
источник
Поскольку ваш тип объекта является индивидуальным, я бы склонен согласиться с вашим решением - разбейте его на более мелкие сегменты, используя метод кодирования (например, JSON или сериализацию контента), а на другом конце - соответствующий код для воссоздания объекта.
источник
Моя версия:
Реализация:
JsonUtils: GitHub
источник
Попробуйте использовать это, у меня это сработало.
источник
Измените типы переменных
private
наpublic
Это просто и удобнее для чтения.
Например
Не работает;
Это работает;
источник
Я сделал хороший вспомогательный класс, который преобразует объект с методами get в массив. Он не полагается на свойства, а только на методы.
Итак, у меня есть следующий объект обзора, который содержит два метода:
Обзор
Комментарий
Написанный мной скрипт преобразует его в массив со свойствами, которые выглядят так:
Источник: сериализатор PHP, который преобразует объект в массив, который можно закодировать в JSON.
Все, что вам нужно сделать, это обернуть json_encode вокруг вывода.
Немного информации о скрипте:
источник
Я потратил несколько часов на решение той же проблемы. Мой объект для преобразования содержит множество других, определения которых я не должен трогать (API), поэтому я придумал решение, которое, я полагаю, может быть медленным, но я использую его для целей разработки.
Преобразует любой объект в массив
Это преобразует любой объект в stdClass
источник
unlink($thisAnswer);
inline php strings
,eval
,shell_exec(php)
... куб.см-комбо.