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;
}