そのため、新しいJsonSerializable Interfaceを見つけたときに、phpオブジェクトをJSONにシリアル化する方法についてphp.netを探していました。それだけだ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)
だからこれはtoArray()
私がMf_Data
クラスのために考案した関数です:
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