因此,当我偶然发现新的 可串行化接口时,我正在 Php.net中寻找关于将 PHP 对象序列化为 JSON 的信息。不过它只是 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)
}
这是我为 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
引用时,它就像一个魔咒。
接下来,转换一个复杂的树节点对象的最后一个函数是:
// 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;
}
我将再次跟进,使用一个更干净的实现。使用接口进行 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'
)));
}
}