Eu estava vagando pelo php.net em busca de informações sobre como serializar objetos PHP para JSON, quando me deparei com a nova interface JsonSerializable . É apenas PHP> = 5.4 , porém, e estou executando em um ambiente 5.3.x.
Como esse tipo de funcionalidade é alcançado em PHP <5.4 ?
Ainda não trabalhei muito com JSON, mas estou tentando oferecer suporte a uma camada de API em um aplicativo, e despejar o objeto de dados ( que de outra forma seria enviado para a visualização ) em JSON seria perfeito.
Se eu tentar serializar o objeto diretamente, ele retorna uma string JSON vazia; o que é porque eu presumo json_encode()
que não saiba o que diabos fazer com o objeto. Devo reduzir recursivamente o objeto em um array e, em seguida, codificá- lo ?
Exemplo
$data = new Mf_Data();
$data->foo->bar['hello'] = 'world';
echo json_encode($data)
produz um objeto vazio:
{}
var_dump($data)
no entanto, funciona como esperado:
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)
}
Termo aditivo
1)
Portanto, esta é a toArray()
função que desenvolvi para a Mf_Data
classe:
public function toArray()
{
$array = (array) $this;
array_walk_recursive($array, function (&$property) {
if ($property instanceof Mf_Data) {
$property = $property->toArray();
}
});
return $array;
}
No entanto, como os Mf_Data
objetos também têm uma referência a seu objeto pai (que contém ), isso falha com a recursão. Funciona perfeitamente quando removo a _parent
referência.
2)
Apenas para acompanhar, a função final para transformar um objeto de nó de árvore complexo que eu escolhi foi:
// 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)
Estou fazendo o acompanhamento novamente, com uma implementação um pouco mais limpa. Usar interfaces para uma instanceof
verificação parece muito mais limpo do que method_exists()
( no entanto method_exists()
, herança / implementação de corte cruzado ).
Usar unset()
parecia um pouco confuso também e parece que a lógica deveria ser refatorada em outro método. No entanto, esta implementação faz copiar a matriz de propriedade ( devido àarray_diff_key
), então algo a considerar.
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