Skip to content

JSON.stringify 递归逻辑研究

Published: at 04:20 PMSuggest Changes

今天突发奇想,想看一下 JSON.stringify 方法得递归逻辑

let data4 = [
  { id: 1,
    children: [
      { id: 2,
        children: [
          { id: 3 ,
          children: [
            { id: 4 },
            { id: 5}]
          }
        ]
      }
    ]
  }
];

JSON.stringify(
  data4,
  (k, v) => {
    if (
      Object.prototype.toString.call(v) === '[object Object]' &&
      'children' in v
    ) {
      const c = v.children;
      console.log('v.id', v.id);
    }
    return v;
  }
);

这样的代码,递归得到 1 2 3 没问题

我做了一个操作,在代码中加入了 v.add = c;,想看看它会怎么样去递归。

调试了一通,发现它的递归逻辑没我想象这么简单,代码如下,注意一下代码中的 v.add = c;

我期望通过 v.add = c; 来猜出它的递归逻辑

let data4 = [
  {
    id: 1,
    children: [
      { id: 2, children: [{ id: 3, children: [{ id: 4 }, { id: 5 }] }] },
    ],
  },
];

JSON.stringify(data4, (k, v) => {
  // console.log('k,v', k, v && JSON.stringify(v));
  if (
    Object.prototype.toString.call(v) === '[object Object]' &&
    'children' in v
  ) {
    const c = v.children;
    console.log('v.id', v.id);
    // 哪位大佬能解释一下加这个代码,理论上会套娃报错,但是却没有报错,这段代码可以注释掉对比执行结果
    v.add = c;
    // 简单拷贝一下也一样不报错
    // v.add = JSON.parse(JSON.stringify(c));
  }
  return v;
});

输出结果是 1 2 3 3 2 3 3

我想知道为什么会这样,在代码里面加了各种打印调试,debugger。

而且我预判会套娃,打算加上这样的代码防止套娃

if (v.add) {
  // 不执行
} else {
  // 没 add 属性再给对象加上 add 属性
  v.add = c;
}

但是结果和我想象的不同,看一下图中的箭头,可以猜出它的递归逻辑,理论上它应该套娃死循环的但是没有。

有个大佬丢了个,V8 的代码 https://v8.github.io/api/head/v8-json_8h_source.html

我打算有时间了看一下

头铁看 V8 源码

src/d8/d8-js.cc src/json/json-stringifier.cc

猜测和这儿有关 To avoid crashing on cyclic objects , 有个递归限制防止奔溃的。

// A more universal stringify that supports more types than JSON.
// Used by the d8 shell to output results.
var stringifyDepthLimit = 4;  // To avoid crashing on cyclic objects

// Hacky solution to circumvent forcing --allow-natives-syntax for d8
function isProxy(o) { return false };
function JSProxyGetTarget(proxy) { };
function JSProxyGetHandler(proxy) { };

Previous Post
ESLint Variable 'data' is already declared in the upper scope. 错误解决
Next Post
GoLand: 调整可编辑文件大小