我有一個節點數組。 每個節點都有一個子數組和一個指向其父節點的指針。 我想序列化它使用JSON.stringify,但與父指針我顯然最終與循環引用,和JSON引發異常。我能做些什麼來解決循環引用和使用JSON序列化?用JSON序列化圖論樹的解決方法?
相關問題:Chrome sendrequest error: TypeError: Converting circular structure to JSON
我有一個節點數組。 每個節點都有一個子數組和一個指向其父節點的指針。 我想序列化它使用JSON.stringify,但與父指針我顯然最終與循環引用,和JSON引發異常。我能做些什麼來解決循環引用和使用JSON序列化?用JSON序列化圖論樹的解決方法?
相關問題:Chrome sendrequest error: TypeError: Converting circular structure to JSON
你應該有家長的對象來創建自定義的toJSON功能。
如果被字符串化的對象有一個名爲的toJSON其值 是功能屬性,則的toJSON方法定製JSON字符串化 行爲
var x = {
foo: 'foo',
toJSON: function() {
return 'bar';
}
};
var json = JSON.stringify({x: x});
所以你可以在具有父引用的對象中創建該函數,可能是這樣的?
MyObj = function(){
this.xxx = 'foobar';
this.zzz = 'foooobar';
this.name = 'foo';
this.parent = ...;
toJSON = function(){
tmp = '{'
for(prop in MyObj){
if(prop == 'parent'){
tmp += 'parent: "'+ this['parent'].name +'"'; //maybe?? optional!
}else{
tmp += prop + ':' + this[prop].stringify + ','; //you will still use the browser function
}
tmp += '}
}
return tmp;
}
}
你還必須寫一個解析函數,或者是照顧的自動? – Joe
解析時你永遠不會遇到循環引用的問題,所以你可以使用預先構建的函數甚至eval函數,它應該沒問題 –
有一個嘗試 - 利用信息和演示數據是在BUTTOM:
// class
var Decircularizer = function() {};
// class contents
Decircularizer.prototype = {
curId: 0x01,
idField: '{`id´}',
_getNewId: function() {
return this.curId++;
},
_getVisited: function(obj) {
return obj[''+this.idField];
},
_setVisited: function(obj) {
return (obj[''+this.idField] = this._getNewId());
},
_removeVisited: function(obj){
delete obj['' + this.idField];
},
decycle: function(obj, depth) {
var $this = this;
depth = depth || 0;
var key = this._getVisited(obj);
if (key > 0x00) return this.idField + key;
if(!jQuery.isPlainObject(obj))
return obj;
key = this._setVisited(obj);
jQuery.each(obj, function(prop, val){
if (prop == $this.idField) return;
if (jQuery.isFunction(obj[prop])) delete obj[prop];
else obj[prop] = $this.decycle(val, depth + 1);
});
return obj;
},
recycle: function(obj){
var $this = this;
var ids = {};
this._searchIds(obj, ids);
return this._recycle(obj, ids);
},
_recycle: function(obj, ids){
var $this = this;
if(!jQuery.isPlainObject(obj))
return obj;
jQuery.each(obj, function(prop, val){
var xval = ids[val];
if(xval)
obj[prop] = xval;
else
obj[prop] = $this._recycle(val, ids);
});
return obj;
},
_searchIds: function(obj, ids){
var $this = this
var ids = ids || {};
var key = this._getVisited(obj);
if(key > 0x00) {
ids[this.idField + key] = obj;
$this._removeVisited(obj);
}
if(!jQuery.isPlainObject(obj))
return ids;
jQuery.each(obj, function(prop, val){
$this._searchIds(val, ids);
});
return ids;
}
};
// EXAMPLE DATA
var a = {};
var a2 = {}
a.b = a;
a.c = "hallo";
a.e = 123;
a.f = a2;
a2.x = a;
// USAGE
// new class
var ser = new Decircularizer();
// uncycle
var cleanObjectWithOutCirculars = ser.decycle(a);
console.debug(cleanObjectWithOutCirculars);
/* object looks like
Object {c: "hallo", e: 123, b: "{`id´}1", {`id´}: 1, f: Object}
b: "{`id´}1"
c: "hallo"
e: 123
f: Object
x: "{`id´}1"
{`id´}: 2
__proto__: Object
{`id´}: 1
__proto__: Object
*/
// recycle
var aNew = ser.recycle(cleanObjectWithOutCirculars);
console.debug(aNew)
/*
Object {c: "hallo", e: 123, b: Object, f: Object}
b: Object
b: Object
c: "hallo"
e: 123
f: Object
x: Object
__proto__: Object
__proto__: Object
c: "hallo"
e: 123
f: Object
x: Object
b: Object
c: "hallo"
e: 123
f: Object
__proto__: Object
__proto__: Object
__proto__: Object
*/
// correct (Y)
我的想法如何處理這一點 - 但給我一些時間... – TheHe