题解 P2386 【放苹果】
ybb756032937 · · 题解
C++代码解析
基本思路:递归 记录 AC
代码呈上:
#include<iostream>//个人不建议使用万能头文件;可能编译出错;
#include<cstdio>
#include<cstdlib>
#include<cmath>
using namespace std;
int sum[10000];//用来记录每一组数据最后的结果
int apple(int m,int n)//m是苹果数,n是盘子数
{
if(m==0||m==1||n==1)//边界:当苹果只有一个或者没有苹果时或者盘子只有一个时,只有一种放法,所以达到边界,返回值;
return 1;
if(m<n)//当苹果数少于盘子数,就只有m个盘子作用,所以接下来就计算m个苹果和m个盘子;
return apple(m,m);
if(m>=n)//如果苹果数大于等于盘子数,分成两个部分,一种是目前所有的盘子都放一个苹果,另一种是拿一个盘子不放;
return apple(m-n,n)+apple(m,n-1);
}
int main()
{
int n,m,s;//其中s是数据的个数;
cin>>s;
for(int i=1;i<=s;i++)
{
cin>>m>>n;//m是苹果数,n是盘子数
sum[i]=apple(m,n);//将每一组的苹果和盘子的到的结果记录在sum数组之中;
}
for(int j=1;j<=s;j++)
cout<<sum[j]<<endl;//输出数据;
return 0;
}
详细解释思路(当苹果数大于等于盘子数时的分法):一种是目前所有的盘子都放一个苹果,盘子数不变,即:apple(m-n,n);
另一种是将一个盘子不放,再来进行思考,即:apple(m,n-1);
两种情况的总和就是答案;
接下来就用一个例子来详细解释:
现在有5个苹果和3个盘子;
因为苹果数大于盘子数,所以分成两种:
第一种:目前所有盘子放一个苹果(目前盘子数3个)apple(2,3);
第二种:拿一个盘子不放苹果,剩下的盘子继续考虑,apple(5,2);
apple(2,3):因为苹果数小于盘子数,所以apple(2,2);
apple(2,2):因为苹果数等于盘子数,所以又分为两种:
第一种:目前所有盘子放一个苹果(目前盘子数2个)apple(0,2); 没有苹果,达到边界,返回值1;
第二种:拿一个盘子不放苹果,剩下的盘子继续考虑,apple(2,1); 盘子数只有1个,达到边界,返回值1;
apple(5,2):因为苹果数大于盘子数,所以又分为两种:
第一种:目前所有盘子放一个苹果(目前盘子数2个)apple(3,2);
第二种:拿一个盘子不放苹果,剩下的盘子继续考虑,apple(5,1); 因为盘子数只有1个,达到边界,返回值1;
apple(3,2):因为苹果数大于盘子数,所以又分为两种:
第一种:目前所有盘子放一个苹果(目前盘子数2个)apple(1,2); 苹果数只有1个,达到边界,返回值1;
第二种:拿一个盘子不放苹果,剩下的盘子继续考虑,apple(3,1); 盘子数只有1个,达到边界,返回值1;
所有的值相加得到最后的结果5,记录在数组sum中,最后输出;
以上就是本题所有的讲解,要查看更多内容,学到更多知识,请查看我的博客:https://www.luogu.org/blog/AHacker/