【Java/算法/排列】求所有仅由1,2,3,4,5组成的没有重复数字的四位数的和

【数学解法】

由1,2,3,4,5组成的没有重复数字的四位数有A54=120个

以千位为例,在此出现的1,2,3,4,5的几率是均等的,即每个数都出现了120/5=24次。

也就是说,120个数的千位加起来是(1*24+2*24+3*24+4*24+5*24)*1000=15*24*1000

同样的道理,120个数的百位加起来是(1*24+2*24+3*24+4*24+5*24)*100=15*24*100

120个数的十位加起来是(1*24+2*24+3*24+4*24+5*24)*10=15*24*10

120个数的个位加起来是(1*24+2*24+3*24+4*24+5*24)*1=15*24*1

其总和=15*24*1111=399960

【代码解法】

做一个A54的排列,把每个数累计即可。

主类Summary:

package test230429;

import java.util.List;

/**
 * 求所有仅由1,2,3,4,5组成的没有重复数字的四位数的和
 */
public class Summary {
    public static void main(String[] args) {
        int[] numbers = {1, 2, 3, 4,5};
        Arranger arranger2 = new Arranger(numbers, 4);

        int sum=0;
        for (List<Integer> re : arranger2.getResults()) {
            sum+=1000*re.get(0)+100*re.get(1)+10*re.get(2)+1*re.get(3);
        }

        System.out.println("Sum="+sum);
    }
}

辅助类Arranger:

package test230429;

import java.util.ArrayList;
import java.util.List;

/**
 * 用于产生排列结果的工具类
 * 从n个元素中取出m个元素,按照一定的顺序排成一列。得到所有排列的方案
 */
public class Arranger {
    // 保存在内部的对原始元素数组的引用
    private int[] arr;

    // 总计多少元素,此即数组长度
    private final int n;

    // 选多少个
    private final int m;

    // 返回结果
    private List<List<Integer>> results;

    /**
     * 构造函数一
     * 这个构造函数是用于全排列的(n=m=数组长度)
     *
     * @arr 原始元素数组
     */
    public Arranger(int[] arr) {
        this.arr = arr;
        this.n = arr.length;
        this.m = arr.length;

        this.results = new ArrayList<>();
        doArrange(new ArrayList<>());
    }

    /**
     * 构造函数二
     * 这个构造函数是用于部分排列的(m<n=数组长度)
     *
     * @param arr    原始元素数组
     * @param selCnt 选多少个
     */
    public Arranger(int[] arr, int selCnt) {
        this.arr = arr;
        this.n = arr.length;
        this.m = selCnt;
        if (m > n) {
            throw new ArrayIndexOutOfBoundsException("m:" + m + " >n:" + n);
        }

        this.results = new ArrayList<>();
        doArrange(new ArrayList<>());
    }

    /**
     * 使用递归进行全排列,结果放在results中
     *
     * @param initialList 初始链表
     */
    private void doArrange(List<Integer> initialList) {
        List<Integer> innerList = new ArrayList<>(initialList);

        if (m == initialList.size()) {
            results.add(innerList);
        }

        for (int i = 0; i < arr.length; i++) {
            if (innerList.contains(arr[i])) {
                continue;
            }

            innerList.add(arr[i]);
            doArrange(innerList);
            innerList.remove(innerList.size() - 1);
        }
    }

    /**
     * 获得结果链表的引用
     *
     * @return
     */
    public List<List<Integer>> getResults() {
        return results;
    }

    // 测试
    public static void main(String[] args) {
        int[] numbers = {1, 2, 3, 4};
        Arranger arranger = new Arranger(numbers);

        System.out.println("四元素全排列示例:");
        int idx = 0;
        for (List<Integer> re : arranger.getResults()) {
            System.out.println(String.format("%02d", ++idx) + "." + re);
        }

        Arranger arranger2 = new Arranger(numbers, 2);
        System.out.println("\n四选二排列示例:");
        idx = 0;
        for (List<Integer> re : arranger2.getResults()) {
            System.out.println(String.format("%02d", ++idx) + "." + re);
        }
    }
}

【程序输出】

Sum=399960

两种分析能相互印证。

END

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 所有由它们组成的无重复数字的3位数如下: - a, a+1, a+2 - a, a+1, a+3 - a, a+1, a+4 - a, a+2, a+3 - a, a+2, a+4 - a, a+3, a+4 - a+1, a+2, a+3 - a+1, a+2, a+4 - a+1, a+3, a+4 - a+2, a+3, a+4 注意:如果a+4大于6,则不在计算范围内。 ### 回答2: 这道题目可以考虑暴力枚举每一个可能的3位数,然后判断这个3位数是否由给定的4个连续整数构成,如果是则输出。 具体来说,我们可以设置两层循环,外层循环枚举所有可能的3位数,即从100到999。而内层循环则枚举所有可能的4个连续整数,即从a到a+3。对于每一个内层循环得到的4个连续整数,我们可以将它们的数字拆分出来,然后用一个set容器来判断这些数字是否有重复。如果没有重复,则说明这4个连续整数可以组成一个无重复数字的3位数,我们就将其输出即可。 下面是具体的代码实现: ```python a = int(input()) # 读入a for i in range(100, 1000): # 枚举所有3位数 num_set = set() # 用set容器来判断是否有重复数字 for j in range(a, a+4): # 枚举所有由a开始的4个连续数字 num_set.update(set(str(j))) # 将这些数字拆分出来,加入到set容器中 if len(num_set) == 3: # 如果set容器中元素个数为3,则说明这4个连续数字可以组成一个无重复数字的3位数 print(i) # 输出这个3位数 ``` 需要注意的是,这个实现方案并不是最优的,因为它的时间复杂度是O(2700*4)=O(1.08万),不过面对只有6个正整数的小数据,这个复杂度是可以接受的。如果需要处理更大的数据,我们可以考虑优化算法,比如一些剪枝策略或者搜索算法。 ### 回答3: 首先,我们需要知道如何计算由给定数字组成的所有无重复数字的3位数。假设给定的数字为a、b、c、d,我们首先将其放入一个数组中,然后使用三重循环。第一重循环枚举百位数字,第二重循环枚举十位数字,第三重循环枚举个位数字。如果三个数字都不相同,则输出它们组成的3位数。 下面是实现该算法的伪代码: int[] digits = {a, b, c, d}; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { for (int k = 0; k < 4; k++) { if (i != j && i != k && j != k) { int number = digits[i] * 100 + digits[j] * 10 + digits[k]; print(number); } } } } 接下来,我们可以将该算法应用于题目中的问题。首先,我们需要枚举给定正整数a开始的连续4个数字。这可以通过一个简单的循环实现。然后,对于每个4个数字,我们使用上述算法获取由它们组成的所有无重复数字的3位数。 下面是实现该算法的伪代码: for (int i = a; i <= a + 3; i++) { int[] digits = {i, i + 1, i + 2, i + 3}; for (int j = 0; j < 4; j++) { for (int k = 0; k < 4; k++) { for (int l = 0; l < 4; l++) { if (j != k && j != l && k != l) { int number = digits[j] * 100 + digits[k] * 10 + digits[l]; print(number); } } } } } 综上所述,我们可以通过枚举连续4个数字,并使用一个三重循环来计算出由它们组成的所有无重复数字的3位数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值