关于N个鸡蛋放在M个篮子里等系列问题详解

2024-04-04 14:38

本文主要是介绍关于N个鸡蛋放在M个篮子里等系列问题详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

N > M。求出满足如下要求的所有鸡蛋方法。要求:1.篮子不能为空;2.对于任意正整数n<=N, 能取x个篮子,使篮子里的鸡蛋数总和等于n。

 

实现方法一:

#include <cstring>
#include <iostream>//namespace al{
int min(int x, int y)
{return x<=y?x:y;
}
void egg (int n, int m, int *&a,int size)
{ // n是鸡蛋数,m是篮子数,a数组用来存储结果,size是a数组的大小for (int i=min((n+1)/2, a[m]); i>=1; --i){if (n<m) {continue;}else if (n == m){for (int j=m-1; j>=0; --j){a[j] = 1;}m = 1;n = 0;}if (m == 1) {if (n > 1) {continue;}else{for (int k=0; k<size; ++k)std::cout<<a[k]<<" ";std::cout<<std::endl;return ;}} else {a[m-1] = i;egg (n-i, m-1, a, size);}}
}
//} //namespace alint main()
{int n=20;int m=10;int* a = new int[m];a[0] = 1;int *&aa = a;for (int i=(n+1)/2; i>=1; --i){ //由于egg函数中有i=min((n+1)/2, a[m])这句,所以数组最后一位是a[m-1],不存在a[m],所以这里初始处理下memset(a, 0, m);a[m-1] = i;//al::egg (n-i, m-1, aa, m);}
}

 

实现方法二:

#include <iostream>   
using namespace std;   
long pow2[20];   
int N,M;   
int ans[1000];   
void solve( int n , int m , int Min )   
{   if(n == N && m == M)   {   for(int i=0;i<M;i++)   {   cout<<ans[i]<<" ";         }    cout<<endl;   return ;         }    else if( n + (M-m)*Min > N || N > pow2[M-m]*n + pow2[M-m]-1)   return ;   else  {   for(int i = Min; i <= n+1; i++)   {   ans[m] =  i;       solve(n+i,m+1,i);    }                  }             
}     
int main()   
{   pow2[0] = 1;   for(int i=1;i<20;i++)   {   pow2[i] = pow2[i-1]<<1;          }   cin>>N>>M;   if( M > N || pow2[M]-1 < N)   {   cout<<"没有有效解"<<endl;               }          solve( 0 , 0 , 1 );   system("pause");       return 0;      
}  


说明:

1.n + (M-m)*Min > N 剪枝条件:放n个鸡蛋后,后面的篮子里即使都放Min个,总鸡蛋数都超过了N个。说明鸡蛋太少了
2.当前篮子放n个鸡蛋,下一个篮子放鸡蛋的个数为Min~n+1,也就是最多放n+1个,再下一个篮子最多放2n+2,4n+4...(n+1)*2^(M-m-1)
   当前篮子放n个,如果以后按最多的放,所有篮子的鸡蛋总和如果小于N,说明鸡蛋太多,放不完,要剪枝。即 
    n+(n+1)(2^0+2^1+2^2+2^3+...+2^(M-m-1))<N
    化简得:
    N > pow2[M-m]*n + pow2[M-m]-1

    此外main函数里的判断pow2[M]-1 < N也是按照这个思路推导的。

实现方法三:

#include <iostream>
using namespace std;#define MAX_M   32
int ar[ MAX_M + 1 ];
int egg = 9 , box = 5;void place_egg( int n , int m , int max )
{if( m == 1 ){ar[ 1 ] = n;for( int i = 1 ; i <= box ; i++ )cout << " " << ar[ i ];cout << endl;return;}if( m > n || n > ( 1 << m ) - 1 )return;if( ( n + 1 ) / 2 < max )max = ( n + 1 ) / 2;for( int i = max ; i >= ( n + m - 1 ) / m ; i-- ){ar[ m ] = i;place_egg( n - i , m - 1 , i );}
}int main()
{place_egg( egg , box , egg );return 0;
}


实现方法四:

/** * 假设 n>m 并且 n小于100 * @author Jason * 2011.3.30 */  
public class Test {  private int m;  private int n;  private int eggs[];  private int numAnswer;  Test(){  m=10;  n=20;  numAnswer=0;  eggs =  new int[m];  for(int i=0;i<m;i++){  eggs[i]=0;  }  }  private void fill(boolean [] state, int step, int sum){  if(step>=m){  state[sum] = true;  return ;  }  fill(state,step+1,sum);  fill(state,step+1,sum+eggs[step]);  }  /** * 判断是否满足:任意一个小于N的正整数,都能由某几个篮子内蛋的数量相加的和得到 * 算法:暴力枚举所有篮子的组合 * @return */  private boolean judge(){  boolean [] state = new boolean [n+1];  for(int i=0;i<=n;i++){  state[i] = false;  }  fill(state,0,0);  for(int i=1;i<=n;i++){  if(!state[i]){  return false;  }  }  return true;  }  /** * 给每个篮子分鸡蛋,升序(后一个篮子的鸡蛋必须不小于前一个篮子,避免重复计算) * @param pre 前一个篮子鸡蛋数 * @param already 前step个篮子 已使用的鸡蛋数 * @param step 第step个篮子 */  public void solve(int pre,int already, int step){  if(step==m-1){  //最后一个篮子   eggs[m-1]=n-already;  //不符合条件   if(eggs[m-1]<pre)    return;  //判断是否满足:任意一个小于N的正整数,都能由某几个篮子内蛋的数量相加的和得到   if(judge()) {  for(int i=0;i<m;i++){  System.out.print(eggs[i]+" ");  }  System.out.println();  numAnswer++;  }  return ;  }  // 给第step个篮子装鸡蛋,pre 到 n-already 种可能   for(int i=pre; i<=n-already; i++){  eggs[step]=i;  //递归   solve(i,already+i,step+1);  }  }  public static void main(String arg []  ){  Test test = new Test();  test.solve(1,0,0);  System.out.println("可能情况的数量:"+test.numAnswer);  }  
}  



 



这篇关于关于N个鸡蛋放在M个篮子里等系列问题详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/876056

相关文章

CSS place-items: center解析与用法详解

《CSSplace-items:center解析与用法详解》place-items:center;是一个强大的CSS简写属性,用于同时控制网格(Grid)和弹性盒(Flexbox)... place-items: center; 是一个强大的 css 简写属性,用于同时控制 网格(Grid) 和 弹性盒(F

spring中的ImportSelector接口示例详解

《spring中的ImportSelector接口示例详解》Spring的ImportSelector接口用于动态选择配置类,实现条件化和模块化配置,关键方法selectImports根据注解信息返回... 目录一、核心作用二、关键方法三、扩展功能四、使用示例五、工作原理六、应用场景七、自定义实现Impor

一文深入详解Python的secrets模块

《一文深入详解Python的secrets模块》在构建涉及用户身份认证、权限管理、加密通信等系统时,开发者最不能忽视的一个问题就是“安全性”,Python在3.6版本中引入了专门面向安全用途的secr... 目录引言一、背景与动机:为什么需要 secrets 模块?二、secrets 模块的核心功能1. 基

一文详解MySQL如何设置自动备份任务

《一文详解MySQL如何设置自动备份任务》设置自动备份任务可以确保你的数据库定期备份,防止数据丢失,下面我们就来详细介绍一下如何使用Bash脚本和Cron任务在Linux系统上设置MySQL数据库的自... 目录1. 编写备份脚本1.1 创建并编辑备份脚本1.2 给予脚本执行权限2. 设置 Cron 任务2

一文详解如何在idea中快速搭建一个Spring Boot项目

《一文详解如何在idea中快速搭建一个SpringBoot项目》IntelliJIDEA作为Java开发者的‌首选IDE‌,深度集成SpringBoot支持,可一键生成项目骨架、智能配置依赖,这篇文... 目录前言1、创建项目名称2、勾选需要的依赖3、在setting中检查maven4、编写数据源5、开启热

Python常用命令提示符使用方法详解

《Python常用命令提示符使用方法详解》在学习python的过程中,我们需要用到命令提示符(CMD)进行环境的配置,:本文主要介绍Python常用命令提示符使用方法的相关资料,文中通过代码介绍的... 目录一、python环境基础命令【Windows】1、检查Python是否安装2、 查看Python的安

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

LiteFlow轻量级工作流引擎使用示例详解

《LiteFlow轻量级工作流引擎使用示例详解》:本文主要介绍LiteFlow是一个灵活、简洁且轻量的工作流引擎,适合用于中小型项目和微服务架构中的流程编排,本文给大家介绍LiteFlow轻量级工... 目录1. LiteFlow 主要特点2. 工作流定义方式3. LiteFlow 流程示例4. LiteF