【发布时间】:2011-10-13 18:37:07
【问题描述】:
所以当我偶然发现新的JsonSerializable Interface 时,我在php.net 中寻找有关将PHP 对象序列化为JSON 的信息。不过它只是 PHP >= 5.4,而且我在 5.3.x 环境中运行。
这种功能是如何实现的PHP ?
我还没有过多地使用 JSON,但我正在尝试在应用程序中支持 API 层,并将数据对象(否则会发送到视图)转储到JSON 将是完美的。
如果我尝试直接序列化对象,它会返回一个空的 JSON 字符串;这是因为我假设json_encode() 不知道该对象到底要做什么。我是否应该递归地将对象简化为一个数组,然后对 that 进行编码?
示例
$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'
)));
}
}
【问题讨论】:
-
+1 好问题,还不知道这个功能。
-
@takeshin - 是的,文档页面上的编辑日期是 4 天前。我很高兴看到它!
-
供其他人参考,json_encode 可以很好地处理对象。但是,它只编码该对象的公共成员。因此,如果您有受保护或私有类变量,那么您需要发布的方法之一或 JsonSerializable。
-
@MatthewHerbst 当然可以。旧问题现在已经过时了,无论如何 JsonSerializable
标签: php object json object-serialization