1

我正在尝试使用 JavaScript,只是为了学习目的,我正在编写一个 forEach 迭代器,它可以遍历嵌套数组或任何其他包含length property.

这是我写的:

var forEach = function(obj, callback, options) {
  var options = options || {};
  var context = options.context || this;    

  if(!isEmpty(obj)) { // isEmpty function just evaluates `return !(!!obj.length);`
    for(var x = 0; x < obj.length; x++) {
      if(!isEmpty(obj[x]) && options.deep === true) {
        forEach.call(context, obj[x], callback, options);
        continue;
      }
      callback.call(context, obj[x]);
    }
  }
};

如果我传递一个嵌套数组,我会得到RangeError: Maximum call stack size exceeded

forEach(['a', 'b', ['c', 'd']], function(x) {
  console.log(x);
}, { deep: true });

但这似乎只有在我检查长度属性时才会发生obj[x]

如果我更换:

if(!isEmpty(obj[x]) && options.deep === true) {

为了:

if((obj[x] instanceof Array) && options.deep === true) {

我会神奇地工作。然而,不仅Arrays有一个长度属性。String有它,所以这不是一个广泛的方法。

我怎样才能防止RangeError但仍然检查length property

编辑:我在NodeJS v0.8.12上运行示例

4

1 回答 1

0

认为那'a'[0][0][0][0][0][0][0]...是无限有效的,并且每个值都是 type string。如果类型是string,那么你不应该递归地迭代它。另请注意,function对象具有length属性,并且function参数可以是对自身的自引用function。这将导致另一个无限递归。我认为以不同的方式处理不同的类型而不是尝试开发一个包罗万象的功能可能更有意义。

您还可以拥有一个maxLevel限制递归深度的属性,其默认值为10. 这样无限递归应该不容易实现。

forEach(['a', 'b', ['c', 'd']], function(x) {
  console.log(x);
}, { deep: true, maxLevel: 10 });

var forEach = function(obj, callback, options, level) {
    var options = options || {};
    var context = options.context || this;

    if (!level) level = 1;
    if (!options.maxLevel) options.maxLevel = 10;

    if (level > options.maxLevel) return;
    ...
        forEach.call(context, obj[x], callback, options, level + 1);
    ...
}

演示:http: //jsfiddle.net/bjpx5/

于 2013-01-22T16:05:34.690 回答