Así que estaba deambulando por php.net en busca de información sobre la serialización de objetos PHP en JSON, cuando me encontré con la nueva interfaz JsonSerializable . Sin embargo, es solo PHP> = 5.4 y estoy ejecutando en un entorno 5.3.x.
¿Cómo se logra este tipo de funcionalidad PHP <5.4 ?
Todavía no he trabajado mucho con JSON, pero estoy tratando de admitir una capa de API en una aplicación, y descargar el objeto de datos ( que de otro modo se enviaría a la vista ) en JSON sería perfecto.
Si intento serializar el objeto directamente, devuelve una cadena JSON vacía; que se debe a que supongo json_encode()
que no sabe qué diablos hacer con el objeto. ¿Debo recursivamente reducir el objeto en una matriz, y luego codificar que ?
Ejemplo
$data = new Mf_Data();
$data->foo->bar['hello'] = 'world';
echo json_encode($data)
produce un objeto vacío:
{}
var_dump($data)
sin embargo, funciona como se esperaba:
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)
}
Apéndice
1)
Entonces esta es la toArray()
función que he ideado para la Mf_Data
clase:
public function toArray()
{
$array = (array) $this;
array_walk_recursive($array, function (&$property) {
if ($property instanceof Mf_Data) {
$property = $property->toArray();
}
});
return $array;
}
Sin embargo, dado que los Mf_Data
objetos también tienen una referencia a su objeto principal ( contenedor ), esto falla con la recursividad. Sin embargo, funciona como un encanto cuando elimino la _parent
referencia.
2)
Solo para continuar, la función final para transformar un objeto complejo de nodo de árbol con el que fui fue:
// 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)
Estoy haciendo un seguimiento de nuevo, con una implementación un poco más limpia. El uso de interfaces para una instanceof
verificación parece mucho más limpio que method_exists()
( sin embargo method_exists()
, la herencia / implementación transversal ).
El uso también unset()
parecía un poco complicado, y parece que la lógica debería refactorizarse en otro método. Sin embargo, esta aplicación hace copiar la matriz de propiedad ( debido aarray_diff_key
), por lo que algo a tener en cuenta.
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