解题思路:
看到方案数,就想到了 dfs ,而且这道题 M 的值在10以内,所以一般情况下是不会超时的,当时思路一定要清晰,这是做题的关键!这是在勉励自己!!!我第一次做的时候,就没有考虑全面,不知道如何处理相同金额的小孩。
注意事项:
参考代码:
#include<iostream>
using namespace std;
int M, N, K;
int arr[15][2];//arr[i][0]存的是 1元 或 2 元 arr[i][1]存的是当前这个小孩有没有被选上,1 选了,0 没选
int vis = 0;//存 方案数
// m 是当前排了几个小孩, n 是 当前的零钱数
void dfs( int m, int sum)
{
if (m == M)//小孩有 M 个时,就计数
{
vis++;
return;
}
for (int i = 1; i <= M; i++)//遍历寻找符合条件的小孩,而且还要保证 他 前面几次都没有被选过
{
if (arr[i][1] == 0)//保证没被选过
{
if (arr[i][0] == 2 && sum > 0)//零钱有剩余就可以选择零钱是 2元 的小孩
{
arr[i][1] = 1;
dfs( m + 1, sum - 1);
arr[i][1] = 0;//回溯
}
if (arr[i][0] == 1)//零钱是 一元 的小孩肯定能选
{
arr[i][1] = 1;
dfs( m + 1, sum + 1);
arr[i][1] = 0;//回溯
}
}
}
}
int main()
{
cin >> M >> N >> K;
for (int i = 1; i <= M; i++)
{
if (i <= N)
arr[i][0] = 1;
else
arr[i][0] = 2;
}
for (int i = 1; i <= M; i++)//这里是挑选 可以作为第一个的小孩,也就是 一元 的小孩做第一个人
{
if (arr[i][0] == 1)
{
arr[i][1] = 1;//选上
dfs(1, 1);
arr[i][1] = 0;//回溯,因为不止一位 一元 的小孩
}
}
cout << vis << endl;
return 0;
}
0.0分
0 人评分
C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:
一点编程也不会写的:零基础C语言学练课程
解决困扰你多年的C语言疑难杂症特性的C语言进阶课程
从零到写出一个爬虫的Python编程课程
只会语法写不出代码?手把手带你写100个编程真题的编程百练课程
信息学奥赛或C++选手的 必学C++课程
蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程
手把手讲解近五年真题的蓝桥杯辅导课程
发表评论 取消回复