我在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)
这是我为Mf_Data
类设计的toArray()
函数:
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
。 - Dan Lugg