有没有办法使“ for循环的for循环”?

时间:2019-06-05 20:11:31

标签: javascript arrays data-structures recursive-datastructures array-algorithms

我正在尝试为自己编程一个扑克计算器,并且我的for循环深达5级。

为此,我将for循环嵌套在一个循环之后。我正在寻找一种简单使用1个循环(或函数)的方法,该方法可以告诉我要深入多少个级别。对于此示例,答案为5,但对于其他示例,答案可能是一个更高(更高)的数字,这将很麻烦。我认为递归是一种实现方式,我只是不知道如何设置(不太了解递归)。谢谢您的帮助,不胜感激。

for(var i=0; i < deck.length; i++){
  for(var j=i+1; j<deck.length; j++){
    for(var k=j+1; k<deck.length;k++){
      for(var m=k+1; m<deck.length;m++){
        for(var n=m+1; n<deck.length;n++){
        combo = deck[i];
        combo += deck[j];
        combo += deck[k];
        combo += deck[m];
        combo += deck[n];
        bighands.push(combo);
        }
      }
    }
  }
}

它有效,我只想要一种更好/更通用的方法。

2 个答案:

答案 0 :(得分:2)

使用生成器可以非常优雅地实现:

  function* loop(depth, start, times, prev = []) {
    if(depth <= 0) {
      yield prev;
     return;
    }

    for(let current = start; current < times; current++) {
      yield* loop(depth - 1, current + 1, times, [...prev, current]);
    }
  }

可用作:

  for(const [j, k, l, m] of loop(4, /*times from*/ 0, /* till*/ 5)) {
    //...
  }

上面的代码将以与for循环相同的方式进行迭代,以确保您可以使用生成器执行更多操作,例如直接生成组合:

  const identity = _ => _;

  function* take(n, of, mapper = identity, prev = []) {
    if(!of.length) return;

    if(prev.length >= n) {
       yield mapper(prev);
       return;
    }

    for(let i = 0; i < of.length; i++) {
       yield* take(n, of slice(i + 1), mapper, [...prev, of[i]]);
    }
 }

 for(const combo of take(4, /*of*/ deck, it => it.reduce((a, b) => a + b))) {
   //...
 }

或者如果您直接需要数组

 function toArray(iterator) {
    let result = [], value;
    while(!({ value } = iterator.next()).done) result.push(value);
    return result;
 }

 const combos = toArray(take(4, /*of*/ deck, it => it.reduce((a, b) => a + b)));

答案 1 :(得分:1)

您可以采用递归方法。

function getHands(array, size, right = []) {
    return array.reduce((r, v, i, a) => {
        var temp = [...right, v];
        if (temp.length === size) {
            r.push(temp.join(' '));
        } else {
            r.push(...getHands(a.slice(i + 1), size, temp));
        }
        return r;
    }, []);
}

var deck = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9'],
    bighands = getHands(deck, 5);

console.log(bighands.length);
console.log(bighands);
.as-console-wrapper { max-height: 100% !important; top: 0; }