蓝桥杯.包子凑数(生成数)

Question:

Solve:

题上要问的是凑不出的数目,如果直接推断哪个数不可能凑出是非常困难的

所以我们考虑反向解决这个问题:

在合适的范围内(0 ~ MAXN)将所有可以凑出的数枚举出来,然后用总数减去凑出的数的数目就是答案了

至于合适的范围这件事情,我是这样想的:

两个数 n 和 m 最大的不可能凑出的数是 n*m-n-m,题上给出的 ai 最大是 100,抛去 n 和 m 不可能同时为 100 这件事情,最大的不可能凑出的数也不会超过 100*100-200 (四舍五入成10000)

在枚举之前还有一个问题,INF

什么情况下会有无数个数无法凑出呢?

样例二的输入 3 2 4 6 其实已经给出了答案,如果所有的 ai 不互质,也就是说它们都是某一个大于1的数的倍数,就会有无数个的数无法凑出,毕竟这样的 ai 所能构造出的数都是它们 gcd 的倍数

枚举正式开始

我们用一个 vis 数组来标记每一个数能否被凑出,首先 0 肯定是可以的,也就是不买嘛....

接下来我们用一个类似动规的思想去思考枚举问题:

如果当前的某个数 j 可以被凑出,那么由这个数与每个 ai 相加的结果 j + ai 也可以被凑出,所以就有了公式

if( vis[ j ] )   vis[ j + a[ i ] ] = true;

有了这个公式以后,思路也就明朗了,对 j  做 0 到 MAXN 的循环, 对 i 做 1 到 n 的循环就 ok 了

Code:

#include <bits/stdc++.h>
using namespace std;
#define N 100
#define MAXN 10000
int n, a[N+1];
bool vis[MAXN+1];
int gcd(int a, int b){
	return b == 0 ? a : gcd(b, a%b);
}
int main(void)
{
	//
	scanf("%d", &n);
	scanf("%d", &a[1]);
	int gcdn = a[1];
	for(int i = 2; i <= n; i++){
		scanf("%d", &a[i]);
		gcdn = gcd(gcdn, a[i]);
	}
	//所有ai不互质,输出INF
	if(gcdn != 1){
		cout <<"INF"; return 0;
	}
	vis[0] = true;
	for(int i = 1; i <= n; i++){
		for(int j = 0; j + a[i] <= MAXN; j++){
			//如果状态j可以构造出,则j+a[i]也可以
			if(vis[j]) vis[j+a[i]] = true;
		}
	}
	//输出结果
	int res = 0;
	for(int i = 1; i <= MAXN; i++){
		if(!vis[i]) res++;
	}
	cout <<res;
  return 0;
}

最后附上蓝桥杯汇总链接:蓝桥杯C/C++A组省赛历年真题题解

声明:图片均来源于蓝桥杯官网,以个人刷题整理为目的,如若侵权,请联系删除~

  • 6
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

UmVfX1BvaW50

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值