由于在这个问题上似乎没有太多活动,这里有一个完全初级/实验性的想法:
<?php
abstract class JSONComplexType extends ArrayObject implements JsonSerializable {
abstract protected function __key($key);
abstract public function jsonSerialize();
public function offsetSet ($key, $newval) {
$offset = $this->__key($key);
if ( is_null($offset) ) {
trigger_error(get_class($this).': invalid key:'.var_export($key, true), E_USER_ERROR);
}
else {
parent::offsetSet($offset, $newval);
}
}
}
class JSONArray extends JSONComplexType {
protected function __key($key) {
if ( is_null($key) ) {
return $this->count();
}
else if ( !is_int($key) && !ctype_digit($key) ) {
return null;
}
else {
$key = intval($key);
return $key <= $this->count() ? $key : null;
}
}
public function jsonSerialize() {
return $this->getArrayCopy();
}
}
class JSONObject extends JSONComplexType {
protected function __key($key) {
return (string)$key;
}
public function jsonSerialize() {
return (object)($this->getArrayCopy());
}
}
function foo($in) {
if ( is_object($in) ) {
$retval = new JSONObject();
foreach( get_object_vars($in) as $key=>$value ) {
$retval[$key] =foo($value);
}
}
else if ( is_array($in) ) {
$retval = new JSONArray();
foreach( array_values($in) as $key=>$value ) {
$retval[$key] = foo($value);
}
}
else {
$retval = $in;
}
return $retval;
}
$in = '{
"A":[
{
"X": 1,
"Y": "z",
"O": {},
"AA":[ {},{},[],[1,2,3],{"I":"V"}]
},
{
"X": 2,
"Y": "Z",
"AA": []
}
]
}';
$json = foo( json_decode($in) );
echo "---- 1 ----\r\n", json_encode($json);
$json['A'][0]['AA'][1]['addObject']='New';
$json['A'][0]['AA'][2][]='New0';
$json['A'][0]['AA'][2][1]='New1';
$json['A'][0]['AA'][2][]='New2';
unset($json['A'][0]['AA'][3][0]);
unset($json['A'][0]['AA'][3][1]);
unset($json['A'][0]['AA'][3][2]);
unset($json['A'][0]['AA'][4]['I']);
echo "\r\n---- 2 ----\r\n", json_encode($json);
打印
---- 1 ----
{"A":[{"X":1,"Y":"z","O":{},"AA":[{},{},[],[1,2,3],{"I":"V"}]},{"X":2,"Y":"Z","AA":[]}]}
---- 2 ----
{"A":[{"X":1,"Y":"z","O":{},"AA":[{},{"addObject":"New"},["New0","New1","New2"],[],{}]},{"X":2,"Y":"Z","AA":[]}]}
输出似乎没问题,但代码需要做很多工作(和文档;-))。这只是我在午休时想出的……