动态规划dp算法经典包子凑数java

简介: 动态规划dp算法经典包子凑数java

目录
题目 包子凑数
动态规划思想
具体代码
__
题目 包子凑数
小明几乎每天早晨都会在一家包子铺吃早餐。他发现这家包子铺有NN种蒸笼,其中第ii种蒸笼恰好能放A_iAi个包子。每种蒸笼都有非常多笼,可以认为是无限笼。

每当有顾客想买XX个包子,卖包子的大叔就会迅速选出若干笼包子来,使得这若干笼中恰好一共有XX个包子。比如一共有33种蒸笼,分别能放3、43、4和55个包子。当顾客想买1111个包子时,大叔就会选22笼33个的再加11笼55个的(也可能选出11笼33个的再加22笼44个的)。

当然有时包子大叔无论如何也凑不出顾客想买的数量。比如一共有33种蒸笼,分别能放4、54、5和66个包子。而顾客想买77个包子时,大叔就凑不出来了。

小明想知道一共有多少种数目是包子大叔凑不出来的。
输入
第一行包含一个整数N。(1 <= N <= 100)
以下N行每行包含一个整数Ai。(1 <= Ai <= 100)
2
4
5
输出
一个整数代表答案。如果凑不出的数目有无限多个,输出INF。
6
动态规划思想
在讲动态规划思想前,本题还使用到了经典数论Ax+By=C(x,y>0)问题:若A,B互质,则有无限个C时的方程无解。否则可能有解。推广到a,b,...,n同时成立。可以利用这个思想求是否有无穷个包子数凑不出来。
所以本题中可以类似的用a1,a2,...an表示能放的包子的个数,找到合适的x,y,.......n凑出C。如果方程有解则能凑出,否则不能凑出。至于求最大公约数可以利用辗转相除法求。
然后本题就可以类似背包问题,使用到一个boolean[] dp,下标index表示index个包子是否能够凑出。当dp[i] 即i个包子可以凑出的话,那么j个包子+第i种蒸笼恰好能放Ai个包子:arr[i]也能够凑出来即index=(j+arr[i])个包子也能够凑出来:dp[j + arr[i]] = true;动态规划的思想就体现在这,另外为了节省时间,在录入数据的同时,就可以对dp[]数据进行更新。
具体代码

  1. import java.util.Scanner;
  2. public class _包子凑数 {
  3. public static void main(String[] args) {
  4. Scanner scanner = new Scanner(System.in);
  5. int n = scanner.nextInt();//N种蒸笼
  6. int yueshu = 0;//最大公约数
  7. int[] arr = new int[n + 1];//以下N行每行包含一个整数Ai
  8. boolean[] dp = new boolean[10000];//下标index表示index个包子是否能够凑出
  9. dp[0] = true;//默认0个包子可以凑出来
  10. //在录入数据的同时,即可对dp[index]index个包子是否能够凑出来进行更新处理。
  11. for (int i = 1; i <= n; i++) {
  12. arr[i] = scanner.nextInt();//录入数据
  13. /**
    • 下面if-else语句动态求输入的数据的最大公约数
    • 求当前第i种蒸笼恰好能放Ai个包子和前一个蒸笼能放的包子个数的最大公约数
  14. */
  15. if (i == 1)
  16. yueshu = arr[1];
  17. else
  18. yueshu = yue(arr[i], yueshu);
  19. //更新dp[]数组
  20. for (int j = 0; j < 10000; j++) {
  21. /**
    • 如果dp[j],即j个包子能够凑出来,
    • 那么j个包子+第i种蒸笼恰好能放Ai个包子:arr[i]也能够凑出来
    • 即index=(j+arr[i])个包子也能够凑出来。
  22. */
  23. if (dp[j])
  24. dp[j + arr[i]] = true;//向后递推,动态规划的体现
  25. }
  26. }
  27. //当最大公约数!=1 说明Ai不互质,则有无限个包子数凑不出来
  28. if (yueshu != 1)
  29. System.out.println("INF");
  30. else {
  31. //否则统计个数
  32. int sum = 0;
  33. for (int i = 0; i < dp.length; i++) {
  34. if (!dp[i])
  35. sum++;
  36. }
  37. System.out.println(sum);
  38. }
  39. }
  40. /**
    • 辗转相除法求a,b两个数的最大公约数
    • @param a
    • @param b
    • @return
  41. */
  42. public static int yue(int a, int b) {
  43. if (b == 0)
  44. return a;
  45. else
  46. return yue(b, a % b);
  47. }
  48. }
相关文章
|
1月前
|
机器学习/深度学习 算法 Java
算法设计(动态规划应用实验报告)实现基于贪婪技术思想的Prim算法、Dijkstra算法
这篇文章介绍了基于贪婪技术思想的Prim算法和Dijkstra算法,包括它们的伪代码描述、Java源代码实现、时间效率分析,并展示了算法的测试用例结果,使读者对贪婪技术及其应用有了更深入的理解。
算法设计(动态规划应用实验报告)实现基于贪婪技术思想的Prim算法、Dijkstra算法
|
1月前
|
算法 Java 测试技术
算法设计(动态规划实验报告) 基于动态规划的背包问题、Warshall算法和Floyd算法
这篇文章介绍了基于动态规划法的三种算法:解决背包问题的递归和自底向上实现、Warshall算法和Floyd算法,并提供了它们的伪代码、Java源代码实现以及时间效率分析。
算法设计(动态规划实验报告) 基于动态规划的背包问题、Warshall算法和Floyd算法
|
1月前
|
算法 Java
LeetCode经典算法题:矩阵中省份数量经典题目+三角形最大周长java多种解法详解
LeetCode经典算法题:矩阵中省份数量经典题目+三角形最大周长java多种解法详解
37 6
|
1月前
|
搜索推荐 算法 Java
经典排序算法之-----选择排序(Java实现)
这篇文章通过Java代码示例详细解释了选择排序算法的实现过程,包括算法的基本思想、核心代码、辅助函数以及测试结果,展示了如何通过选择排序对数组进行升序排列。
经典排序算法之-----选择排序(Java实现)
|
1月前
|
搜索推荐 算法 Java
|
1月前
|
存储 算法 Java
LeetCode经典算法题:打家劫舍java详解
LeetCode经典算法题:打家劫舍java详解
46 2
|
1月前
|
人工智能 算法 Java
LeetCode经典算法题:井字游戏+优势洗牌+Dota2参议院java解法
LeetCode经典算法题:井字游戏+优势洗牌+Dota2参议院java解法
37 1
|
1月前
|
存储 算法 Java
LeetCode经典算法题:预测赢家+香槟塔java解法
LeetCode经典算法题:预测赢家+香槟塔java解法
35 1
|
1月前
|
算法 Java
HanLP — HMM隐马尔可夫模型 -- 维特比(Viterbi)算法 --示例代码 - Java
HanLP — HMM隐马尔可夫模型 -- 维特比(Viterbi)算法 --示例代码 - Java
29 0
|
1月前
|
搜索推荐 算法 Java
插入排序算法(Java代码实现)
这篇文章通过Java代码示例详细解释了插入排序算法的实现过程,包括算法的基本思想、核心代码、辅助函数以及测试结果,展示了如何通过插入排序对数组进行升序排列。