因此,当我在php.net周围寻找有关将对象序列化为JSON的信息时,我偶然发现了新的JsonSerializable Interface。不过,它只是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'
)));
}
}
发布于 2011-07-27 05:09:17
编辑:目前是2016-09-24,PHP5.4已经发布了2012-03-01,支持已经结束了2015-09-01。尽管如此,这个答案似乎获得了好评。如果你仍然使用低于5.4版本的PHP,你就会带来安全风险,并使你的项目陷入困境。如果您没有令人信服的理由继续使用低于5.4的版本,或者甚至已经在使用>= 5.4版本,那么就不要使用这个答案,而只是使用PHP>= 5.4(或者,您知道,最近的版本)并实现the JsonSerializable interface
您可以定义一个函数,例如名为getJsonData();
的函数,该函数将返回一个数组、stdClass
对象或其他具有可见参数的对象,而不是私有/受保护的参数,并执行json_encode($data->getJsonData());
。本质上,实现5.4中的函数,但手动调用它。
这样就行了,因为get_object_vars()
是从类内部调用的,可以访问私有/受保护的变量:
function getJsonData(){
$var = get_object_vars($this);
foreach ($var as &$value) {
if (is_object($value) && method_exists($value,'getJsonData')) {
$value = $value->getJsonData();
}
}
return $var;
}
发布于 2011-07-27 05:43:33
在最简单的情况下,类型提示应该是有效的:
$json = json_encode( (array)$object );
发布于 2011-07-27 05:14:58
json_encode()
将只对公共成员变量进行编码。因此,如果你想包括私人,你必须自己做(正如其他人所建议的)。
https://stackoverflow.com/questions/6836592
复制相似问题