如何在JavaScript中创建计数生成器/迭代器级联

How to create a counting generator/iterator cascade in JavaScript?

本文关键字:级联 迭代器 JavaScript 创建      更新时间:2023-09-26

新版本的JavaScript允许将生成器/迭代器与yield关键字结合使用。


我的问题背景

考虑下面的生成器,它产生"0 ~ 9的数字(数字):

// generator (produces numbers from 0 to 9, then stops)
function *zcounter() {
  var i = 0;
  while (i<=9) {
    yield i;
    i++;
  }
}

现在我想用它来替换下面的函数,它使用3个嵌套的for循环:

// e.g.: var iArray=[0,0,0];
function IterateCascade1(iArray) {
    var iterations=0;
    for (var x=0; x<=9; x++) {
        iArray[0]=x;
        for (var y=0; y<=9; y++) {
            iArray[1]=y;
            for (var z=0; z<=9; z++) {
                iArray[2]=z;
                logArray(iArray);
                iterations++;
            }
        }
    }
    return iterations;
}

如果你像

那样调用上面的函数
console.log("Iterations: "+IterateCascade1([0,0,0]));

然后它将从000到999数1000次,这正是我想要的。

缺点是只能使用包含3个元素的数组,不能传递包含更多元素的数组。


通过使用生成器zcounter()来解决它,我尝试了以下操作:

// e.g.: var iArray=[0,0,0];
function IterateCascade2(iArray) {
    var iterations=0;
    // generate 3 iterators
    var gArray = [];
    for(var i=0; i<iArray.length; i++) {    
        var g=zcounter();
        gArray[i]=g;
    }
    // loop through
    for(var a in gArray) {
        //console.log("a:"+a);
        var g=gArray[a];
        var gnext=g.next();
        while (!gnext.done)
        {
            iArray[a]=gnext.value;
            logArray(iArray);
            gnext=g.next();
            iterations++;
        }
    }
    return iterations;
}

如果你像

那样调用上面的函数
console.log("Iterations: "+IterateCascade2([0,0,0]));

那么它将只计数30次,并且它不会像IterateCascade1那样遍历所有1000个数字。

此外,如果传递像

这样的大数组
console.log("Iterations: "+IterateCascade2([0,0,0,0]));

则会"count"从0到9的每一个数字,但不会经过所有10000个组合。


我知道这里缺少一个递归。

  • 如何修改IterateCascade2,使它做正确的事情(循环通过所有组合,你可以传递任何大小的整数数组)?

注意:

为了显示组合,我使用了

function logArray(x) {
    var result="";
    for(var i=0; i<x.length; i++) { result += x[i].toString(); }
    console.log(result);
}

在上面的例子中。您可以使用3种浏览器中的任何一种开发人员工具或JSShell来运行代码。

这是一个递归的解决方案

function iterareIndex(index, array, iterator) {
    var previousValue = array[index];
    for (var i = 0; i < 10; ++i) {
        array[index] = i;
        if (index == array.length - 1) {
            iterator(array);
        } else {
            iterareIndex(index + 1, array, iterator);
        }
    }
    array[index] = previousValue;
}
function IterateCascade2(array){
    iterareIndex(0, array, logArray);
}

顺便说一下,这将给你相同的输出,可能更短:

var size = 3;
var prefix = '';
for(var i = 0; i < size; ++i) {
    prefix += '0';
}
for(var i = 0, l = Math.pow(10, 3); i < l; ++i){
    var printableValue = (prefix+i).slice(-size);
    console.log(printableValue);
}