生成带重复的笛卡尔乘积过程 Cartesian Product with Repetition

目录

  • [What is Cartesian Product with Repetition](#What is Cartesian Product with Repetition)
  • [Code Demo](#Code Demo)

What is Cartesian Product with Repetition

比如说有两个集合:
\(\{1, 2, 3\}\)
\(\{A, B, C\}\)

想把他们组合成所有可能组合,比如,

1AAA

1AAB

1AAC

...

这种组合可以称为"有重复的笛卡尔积"或"带重复的笛卡尔乘积"(Cartesian Product with Repetition)。

带重复的笛卡尔乘积(Cartesian Product with Repetition)具有以下特征:

  • 涉及两个或多个集合:它是针对两个或更多个集合进行操作。在您的示例中,涉及两个集合{1, 2, 3}和{A, B, C}。

  • 元素可以重复出现:与传统的笛卡尔积不同,在带重复的笛卡尔乘积中,同一个集合中的元素可以在生成的组合中多次出现。在您的示例中,第二个集合{A, B, C}中的元素可以在组合中重复出现,比如AAA、ABB等。

  • 生成的组合长度固定:生成的组合的长度是事先确定的,等于参与集合的数量。在您的示例中,生成的组合长度为2,因为涉及两个集合。

  • 组合的排列顺序不同视为不同组合:生成的组合中,元素的排列顺序不同就被视为不同的组合。例如,AB和BA是两个不同的组合。

  • 组合数量呈指数级增长:随着参与集合的元素数量增加,生成的组合数量会呈指数级增长。如果有n个集合,每个集合有m个元素,那么生成的组合数量将是m^n。

总的来说,带重复的笛卡尔乘积是一种特殊的组合形式,它允许元素重复出现。

Code Demo

java 复制代码
import java.util.ArrayList;
import java.util.List;

public class Combination {
    public static void main(String[] args) {
        List<String> list1 = new ArrayList<>();
        list1.add("1");
        list1.add("2");
        list1.add("3");

        List<String> list2 = new ArrayList<>();
        list2.add("A");
        list2.add("B");
        list2.add("C");

        // 用于存储所有可能的组合
        List<String> combinations = new ArrayList<>();

        // 遍历第一个列表中的每个元素
        // 遍历list1中的每个元素,对于每个元素:
        //                      调用generateCombinations方法,将list2和list2的长度作为参数传递进去,并将生成的组合添加到combinations列表中。
        for (String item1 : list1) {
            // 对于每个元素,生成第二个列表中元素的所有可能组合,并添加到 combinations 列表中
            combinations.addAll(generateCombinations(list2, list2.size(), item1));
        }

        // 打印所有可能的组合
        System.out.println("所有可能的组合:");
        for (String combination : combinations) {
            System.out.println(combination);
        }
    }

    /**
     * 生成第二个列表中元素的所有可能组合
     *
     * @param list    第二个列表
     * @param length  要生成的组合长度
     * @param prefix  已经选择的元素前缀
     * @return 所有可能的组合
     */
    private static List<String> generateCombinations(List<String> list, int length, String prefix) {
        // 所有可能的组合 存储
        List<String> combinations = new ArrayList<>();

        // 如果要生成的组合长度为 0,说明已经生成了所有元素的组合
        if (length == 0) {
            // 将前缀添加到结果列表中并返回
            combinations.add(prefix);
            return combinations;
        }

        // 遍历第二个列表中的每个元素
        for (int i = 0; i < list.size(); i++) {
            // 将当前元素追加到前缀中
            String newPrefix = prefix + list.get(i);
            // 递归调用 generateCombinations 方法,将 length 减 1,并将新的前缀作为参数传递进去
            combinations.addAll(generateCombinations(list, length - 1, newPrefix));
        }

        return combinations;
    }
}

out:

所有可能的组合:
1AAA
1AAB
1AAC
1ABA
1ABB
1ABC
1ACA
1ACB
1ACC
1BAA
1BAB
1BAC
1BBA
1BBB
1BBC
1BCA
1BCB
1BCC
1CAA
1CAB
1CAC
1CBA
1CBB
1CBC
1CCA
1CCB
1CCC
2AAA
2AAB
2AAC
2ABA
2ABB
2ABC
2ACA
2ACB
2ACC
2BAA
2BAB
2BAC
2BBA
2BBB
2BBC
2BCA
2BCB
2BCC
2CAA
2CAB
2CAC
2CBA
2CBB
2CBC
2CCA
2CCB
2CCC
3AAA
3AAB
3AAC
3ABA
3ABB
3ABC
3ACA
3ACB
3ACC
3BAA
3BAB
3BAC
3BBA
3BBB
3BBC
3BCA
3BCB
3BCC
3CAA
3CAB
3CAC
3CBA
3CBB
3CBC
3CCA
3CCB
3CCC
相关推荐
轩源源1 个月前
C++草原三剑客之一:继承
开发语言·数据结构·c++·算法·青少年编程·继承·组合
闻缺陷则喜何志丹3 个月前
【C++ 滑动窗口】2134. 最少交换次数来组合所有的 1 II
c++·算法·leetcode·力扣·交换·组合·最少
Betty’s Sweet7 个月前
C++必修:深入理解继承与虚继承
c++·继承·单继承·菱形继承·组合·虚继承
酒茶白开水7 个月前
SwiftUI六组合复杂用户界面
ui·swiftui·组合·复杂界面
DieSnowK8 个月前
[Algorithm][回溯][组合][目标和][组合总和]详细讲解
算法·leetcode·深度优先·回溯·组合·目标和·组合总和
Aaron_Yao_Aloe1 年前
关于卡特兰数
数学·组合
小宇成长录1 年前
C++:继承
c++·笔记·继承·菱形继承·组合
yangqin@12251 年前
【华为OD题库-024】组装最大可靠性设备-java
java·华为od·组合