Więc byłem wędrówki php.net informacji na temat szeregowania PHP obiektów do JSON, gdy natknąłem się nowym interfejsem JsonSerializable . Jest to jednak tylko PHP> = 5.4 i pracuję w środowisku 5.3.x.
W jaki sposób osiągnięto taką funkcjonalność PHP <5.4 ?
Nie pracowałem jeszcze zbyt wiele z JSON, ale próbuję obsługiwać warstwę API w aplikacji i zrzucenie obiektu danych ( który w przeciwnym razie zostałby wysłany do widoku ) do JSON byłoby idealne.
Jeśli spróbuję bezpośrednio serializować obiekt, zwraca on pusty ciąg JSON; a to dlatego, że zakładam, json_encode()
że nie wie, co do cholery zrobić z obiektem. Powinienem rekurencyjnie zmniejszyć obiekt do tablicy, a następnie zakodować , że ?
Przykład
$data = new Mf_Data();
$data->foo->bar['hello'] = 'world';
echo json_encode($data)
tworzy pusty obiekt:
{}
var_dump($data)
jednak działa zgodnie z oczekiwaniami:
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)
}
Uzupełnienie
1)
Oto toArray()
funkcja, którą wymyśliłem dla Mf_Data
klasy:
public function toArray()
{
$array = (array) $this;
array_walk_recursive($array, function (&$property) {
if ($property instanceof Mf_Data) {
$property = $property->toArray();
}
});
return $array;
}
Jednak ponieważ Mf_Data
obiekty mają również odniesienie do ich obiektu nadrzędnego ( zawierającego ), kończy się to niepowodzeniem w przypadku rekursji. Działa jak urok, kiedy usuwam _parent
odniesienie.
2)
Aby kontynuować, ostatnią funkcją do przekształcenia złożonego obiektu z węzłem drzewa, z którym poszedłem, było:
// 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)
Kontynuuję ponownie, z nieco czystszą implementacją. Używanie interfejsów do instanceof
sprawdzenia wydaje się znacznie czystsze niż method_exists()
( jednak method_exists()
dziedziczenie / implementacja krzyżowa ).
Używanie również unset()
wydawało się nieco kłopotliwe i wydaje się, że logikę należy przełożyć na inną metodę. Jednak ta implementacja ma skopiować tablicę posesji ( z powoduarray_diff_key
), więc coś do rozważenia.
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'
)));
}
}