获得笛卡尔积的算法

Algorithm to get Cartesian product

我有一个像 [0,2,3,0,1] 这样的数组作为输入,我需要找到 {0}x{0,1,2}x{0,1,2,3}x{0}x{0,1} 的笛卡尔积,更准确地说,我需要如下输出。


输入:

[0, 2, 3, 0, 1]

输出:

[0, 0, 0, 0, 0]
[0, 0, 0, 0, 1]
[0, 0, 1, 0, 0]
[0, 0, 1, 0, 1]
[0, 0, 2, 0, 0]
[0, 0, 2, 0, 1]
[0, 0, 3, 0, 0]
[0, 0, 3, 0, 1]
[0, 1, 0, 0, 0]
[0, 1, 0, 0, 1]
[0, 1, 1, 0, 0]
[0, 1, 1, 0, 1]
[0, 1, 2, 0, 0]
[0, 1, 2, 0, 1]
[0, 1, 3, 0, 0]
[0, 1, 3, 0, 1]
[0, 2, 0, 0, 0]
[0, 2, 0, 0, 1]
[0, 2, 1, 0, 0]
[0, 2, 1, 0, 1]
[0, 2, 2, 0, 0]
[0, 2, 2, 0, 1]
[0, 2, 3, 0, 0]
[0, 2, 3, 0, 1]

我需要一个通用算法。任何的想法 ?我想用 C++ 编写它。 谢谢

硬代码解决方案是:

for (int a1 : {0}) {
  for (int a2 : {0,1,2}) {
    for (int a3 : {0,1,2,3}) {
      for (int a4 : {0}) {
        for (int a5 : {0,1}) {
            do_job(a1, a2, a3, a4, a5);
        }
      }
    }
  }
}

您可以使用以下通用方法(将所有 a 放入向量中):

bool increase(const std::vector<std::size_t>& v, std::vector<std::size_t>& it)
{
    for (std::size_t i = 0, size = it.size(); i != size; ++i) {
        const std::size_t index = size - 1 - i;
        ++it[index];
        if (it[index] > v[index]) {
            it[index] = 0;
        } else {
            return true;
        }
    }
    return false;
}

void iterate(const std::vector<std::size_t>& v)
{
    std::vector<std::size_t> it(v.size(), 0);

    do {
        do_job(it);
    } while (increase(v, it));
}

Live Demo

从你的例子来看,你似乎想要混合基数,你输入的是每个位置的最大数字值。

一个简单的方法是使用算术编码,特殊处理最大数字0的位置。这是没有 0 最大数字的情况的伪代码:

input radixes
let n_numbers = product_of radixes
for i = 0 to n_numbers-1 inclusive
    let n = i
    for digit_pos = 0 to number-of-radixes-1
        output n mod radix[digit_pos]
        let n = n div radix[digit_pos]
    output newline

我将 0 作为最大数字的处理留在一个位置,作为练习。 :)


我想不起在 C++ 标准库中对此有任何特别相关的支持。 IE。它主要是一个独立于语言的问题,与 C++ 无关,与排列无关,与数组无关。如果我对它的解释是正确的:如果问题也用文字描述,而不仅仅是一个例子,那就更好了。

考虑到您可能不知道输入向量的长度,递归似乎是解决此问题的最简单方法。像这样:

import java.util.ArrayList;
import java.util.Arrays;

public class Cartesian {

  public static void printCartesian(String head, ArrayList<Integer> array) {
    if (array.size() == 0) {
      System.out.println(head);
      return;
    }
    if (array.size() == 1) {
      for (int i = 0; i <= array.get(0); ++i) {
        System.out.println(head + i + "]");
      }
      return;
    }
    // assert array.size() > 1
    int a0 = array.get(0);
    ArrayList<Integer> array0 = new ArrayList<Integer>(array);
    array0.remove(0);
    for (int i = 0; i <= a0; ++i) {
      printCartesian(head + i + ", ", array0);
    }
  }

  public static void main(String[] args) {
    Integer[] input = { 0, 2, 3, 0, 1 };
    ArrayList<Integer> array = new ArrayList<Integer>(Arrays.asList(input));
    printCartesian("[", array);
  }
}

初始调用将是 printCartesian("[", a),其中 a 是一个按顺序包含数组元素的 ArrayList。必须为 size == 1 添加一个额外的 if 子句,否则会打印太多逗号。

将其转换为 C++ 数据结构应该不难。