Тому я блукав по 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