热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

N个鸡蛋放进M个篮子问题

题目:有N个鸡蛋和M个篮子,把鸡蛋放到M个篮子里,每个篮子都不能为空。另外,需要满足:任意一个小于N的正整数,都能由某几个篮子内蛋的数量相加的和得到。写出程序,使得输入一个(N,M),输出所有可能的分

题目:
有N个鸡蛋和M个篮子,把鸡蛋放到M个篮子里,每个篮子都不能为空。另外,需要满足:任意一个小于N的正整数,都能由某几个篮子内蛋的数量相加的和得到。写出程序,使得输入一个(N,M),输出所有可能的分配情况。

 

从题意中应该可以得出,对于(1,1,2,2)和(1,2,1,2)这两种组合,应该是一样的。
因而对于这M个篮子中的鸡蛋数量,我们用数组basket[M]来表示,我们按照非递减顺序进行排列,即basket[i] <= basket[i+1]

1.我们利用归纳法来总结出一个规律:
   对于前n个篮子,其鸡蛋数量总和为Sn,那么对于第n+1个篮子,其鸡蛋数量应该满足:
   basket[n+1] <= Sn + 1,如果basket[n+1] > Sn + 1,那么Sn + 1这个数将无法通过相应的篮子鸡蛋数相加来获得。
   由于是非递减序列,因而
   basket[n] <= basket[n+1] <= Sn + 1

 

2.我们来证明符合上式的数组能够满足条件“任意一个小于N的正整数,都能由某几个篮子内蛋的数量相加的和得到”。
   当M = 1时,basket[0] = 1,当M=2时,取basket[1] = 1,能够满足上述条件。
                                                         取basket[1] = 2,也能够满足上述条件。
   假设M = n-1时,满足上述条件,我们来证明当M = n时亦满足。
   前n-1个篮子的鸡蛋数量总和为Sn-1,此时加上第n个篮子,总和为Sn = Sn-1 + basket[n-1]。即证明Sn - 1,Sn - 2,Sn - 3,Sn - (basket[n-1] - 1)都可以由某几个篮子内蛋的数量相加的和得到。由于basket[n-1] <= Sn-1,而且小于或者等于Sn-1的数能由某几个篮子内蛋的数量相加的和得到,所以Sn - 1,Sn - 2,Sn - 3,Sn - (basket[n-1] - 1)亦可得到。
   证毕。

3.对于N和M的值,我们在输入后即可做一个判断。
   2.1  当N    2.2  当N >= M时,第一个篮子必然要放1个鸡蛋,其后面的篮子我们按照basket[n] <= basket[n+1] <= Sn + 1取最大值,依次为2,4,8,16......,鸡蛋总数为2^M - 1,即M个篮子的鸡蛋数量最大值。

   所以M <= N <2^M

 

4.代码要点

   4.1 对于函数

void solve(int current_sum, int basket_id, int current_num, int* basket, int N, int M)

   其中current_sum表示当前所有篮子鸡蛋的总和,

         basket_id表示当前篮子的序号,
         current_num表示将要放到当前篮子去的鸡蛋数量,
         basket, N, M值都是main函数中的原值,在递归中这三个参数基本没变。
   初始化为(0, 0, 1, basket, N, M)表示此时所有鸡蛋数量为0,将要把1个鸡蛋放进第0个篮子里面。

   4.2 对于函数solve中的

    if ((current_sum + current_num*(M - basket_id)) > N || (current_sum + (current_sum + 1)*((1<<(M - basket_id)) - 1)) < N)
return;

         我们采用的是搜索中的剪枝技术,即在每次递归时,通过预先判断来看此路是否走得通。

  (current_sum + current_num*(M - basket_id)) > N 表示之后的所有篮子都添加最小鸡蛋数量,如果这都大于N,那么肯定不符。
  (current_sum + (current_sum + 1)*((1<<(M - basket_id)) - 1))  

        其中(current_sum + 1)*((1<<(M - basket_id)) - 1) 可以这样解释:

        假设前面的篮子总和为n,那么紧挨着的后一个篮子里鸡蛋数量最大值为n+1,其后的一个篮子最大值为n + (n + 1) + 1 = 2n + 2,这之后的一个篮子的最大值为n + (n + 1) + (2n + 2) + 1 = 4n + 4......(即这里取的都是S+ 1)
        依次类推,我们发现n + 1 + (2n + 2) + (4n + 4) + ...... = (2^count - 1)*(n + 1),count表示相应的篮子数量。

 

5.代码如下:

N eggs M baskets
 1 #include 
2 #include
3
4 void solve(int current_sum, int basket_id, int current_num, int* basket, int N, int M)
5 {
6 if (current_sum == N && basket_id == M)
7 {
8 int i;
9 for (i = 0; i )
10 printf("%d\t", basket[i]);
11 printf("\n");
12 return;
13 }
14
15 if (current_num > N || basket_id >= M)
16 return;
17
18 if ((current_sum + current_num*(M - basket_id)) > N ||
19 (current_sum + (current_sum + 1)*((1<<(M - basket_id)) - 1)) < N)
20 return;
21
22 int j;
23 for (j = current_num; j <= current_sum + 1; j++)
24 {
25 basket[basket_id] = j;
26 solve(current_sum + j, basket_id + 1, j, basket, N, M);
27 }
28 }
29
30 int main()
31 {
32 int N;//the number of eggs
33 int M;//the number of baskets
34 while (scanf("%d%d", &N, &M) != EOF)
35 {
36 if (N = 1<0)
37 printf("Wrong data!\n");
38 else
39 printf("The combinations are as below:\n");
40
41 int* basket = (int*)malloc(sizeof(int)*M);
42 solve(0, 0, 1, basket, N, M);
43 free(basket);
44 }
45 return 0;
46 }

 

 1 #include 
2
3 void solve(int current_sum, int basket_id, int current_num, int* basket, int N, int M)
4 {
5 if (current_sum == N && basket_id == M)
6 {
7 int i;
8 for (i = 0; i )
9 printf("%d\t", basket[i]);
10 printf("\n");
11 return;
12 }
13
14 if (current_num > N || basket_id >= M)
15 return;
16
17 if ((current_sum + current_num*(M - basket_id)) > N ||
18 (current_sum + (current_sum + 1)*((1<<(M - basket_id)) - 1)) < N)
19 return;
20
21 int j;
22 for (j = current_num; j <= current_sum + 1; j++)
23 {
24 basket[basket_id] = j;
25 solve(current_sum + j, basket_id + 1, j, basket, N, M);
26 }
27 }
28
29 int main()
30 {
31 int N;//the number of eggs
32 int M;//the number of baskets
33 while (scanf("%d%d", &N, &M) != EOF)
34 {
35 if (N = 1<0)
36 printf("Wrong data!\n");
37 else
38 printf("The combinations are as below:\n");
39
40 int* basket = new int[M];
41 solve(0, 0, 1, basket, N, M);
42 delete[] basket;
43 }
44 return 0;
45 }

推荐阅读
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • 本文为Codeforces 1294A题目的解析,主要讨论了Collecting Coins整除+不整除问题。文章详细介绍了题目的背景和要求,并给出了解题思路和代码实现。同时提供了在线测评地址和相关参考链接。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文介绍了C++中省略号类型和参数个数不确定函数参数的使用方法,并提供了一个范例。通过宏定义的方式,可以方便地处理不定参数的情况。文章中给出了具体的代码实现,并对代码进行了解释和说明。这对于需要处理不定参数的情况的程序员来说,是一个很有用的参考资料。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • c语言\n不换行,c语言printf不换行
    本文目录一览:1、C语言不换行输入2、c语言的 ... [详细]
  • 本文介绍了一种划分和计数油田地块的方法。根据给定的条件,通过遍历和DFS算法,将符合条件的地块标记为不符合条件的地块,并进行计数。同时,还介绍了如何判断点是否在给定范围内的方法。 ... [详细]
  • 本文介绍了解决二叉树层序创建问题的方法。通过使用队列结构体和二叉树结构体,实现了入队和出队操作,并提供了判断队列是否为空的函数。详细介绍了解决该问题的步骤和流程。 ... [详细]
  • 本文介绍了C函数ispunct()的用法及示例代码。ispunct()函数用于检查传递的字符是否是标点符号,如果是标点符号则返回非零值,否则返回零。示例代码演示了如何使用ispunct()函数来判断字符是否为标点符号。 ... [详细]
  • 本文介绍了UVALive6575题目Odd and Even Zeroes的解法,使用了数位dp和找规律的方法。阶乘的定义和性质被介绍,并给出了一些例子。其中,部分阶乘的尾零个数为奇数,部分为偶数。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
author-avatar
小森林
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有