题目描述
在网友的国度中共有 𝑛 种不同面额的货币,第 𝑖种货币的面额为 𝑎[𝑖],你可以假设每一种货币都有无穷多张。为了方便,我们把货币种数为 𝑛n、面额数组为 𝑎[1..𝑛] 的货币系统记作 (𝑛,𝑎)。
在一个完善的货币系统中,每一个非负整数的金额 𝑥 都应该可以被表示出,即对每一个非负整数 𝑥,都存在 𝑛 个非负整数 𝑡[𝑖] 满足 𝑎[𝑖]×𝑡[𝑖] 的和为 𝑥。然而, 在网友的国度中,货币系统可能是不完善的,即可能存在金额 𝑥不能被该货币系统表示出。例如在货币系统 𝑛=3, 𝑎=[2,5,9]中,金额 1,3 就无法被表示出来。
两个货币系统 (𝑛,𝑎) 和 (𝑚,𝑏)是等价的,当且仅当对于任意非负整数 𝑥,它要么均可以被两个货币系统表出,要么不能被其中任何一个表出。
现在网友们打算简化一下货币系统。他们希望找到一个货币系统 (𝑚,𝑏),满足 (𝑚,𝑏) 与原来的货币系统 (𝑛,𝑎) 等价,且 𝑚 尽可能的小。他们希望你来协助完成这个艰巨的任务:找到最小的 𝑚。
输入格式
输入文件的第一行包含一个整数 𝑇,表示数据的组数。
接下来按照如下格式分别给出 𝑇 组数据。 每组数据的第一行包含一个正整数 𝑛。接下来一行包含 𝑛 个由空格隔开的正整数 𝑎[𝑖]。
输出格式
输出文件共有 𝑇 行,对于每组数据,输出一行一个正整数,表示所有与 (𝑛,𝑎) 等价的货币系统 (𝑚,𝑏) 中,最小的 𝑚。
样例 #1
样例输入 #1
2 4 3 19 10 6 5 11 29 13 19 17
Copy
样例输出 #1
2 5
Copy
提示
在第一组数据中,货币系统 (2,[3,10]) 和给出的货币系统 (𝑛,𝑎)等价,并可以验证不存在 𝑚<2的等价的货币系统,因此答案为 2。 在第二组数据中,可以验证不存在 𝑚<𝑛 的等价的货币系统,因此答案为 5。
【数据范围与约定】
对于 100% 的数据,满足 1≤𝑇≤20,𝑛,𝑎[𝑖]≥1。
解题思路
这道题转了一圈,发现都写得好复杂,其实思路就一个核心,因为题目需要A集合中不能表示的数,B集合中也不能表示,所以B集合里的数,必定是在A集合中的,最大就是大不了跟A集合一模一样,所以这道题我们可以转换成这样一个问题:
现在给你一个A集合,让你优化A集合个数,如果集合内的其他数可以组成ai,就可将ai优化
(灰太狼:我还会回来的……)(作者语:一边去),否则,这个数就是无敌的(哈哈哈哈……),问,将A集合优化后,产生一个B集合,B集合内数字有多上个?
我们将问题简化后,就很好解决了,毕竟数据不大,我们直接将每一个ai与其他数的和遍历一遍。
双重循环就可以解决,i表示A集合第i个,j表示遍历的和。
我们就只要判断j-a[i]的位置是不是初始的0,如果不是,就A集合个数减1,再将a[i]与其他数的和设为1,以便下一次判断。
循环写出来是这样的
dp[0]=1;
for(int i=1;i<=n;i++){
if(dp[a[i]]){
ans--;
continue;
}
for(int j=a[i];j<=a[n];j++){
if(dp[j-a[i]])dp[j]=dp[j-a[i]];//这里dp[j-a[i]]必为1,所以不用再去判断是几,就1、0两种情况
}
}
于是这道“特别特别特别”难的题就写出来了,我们简化后发现就是一道最简单的动态规划,出题者费尽心机给他加了那么多的“废话”,真是太“敬业”,太“良心”了,我都忍不住要送上亲切的问候了……
不说废话了,上代码
代码实现
#include <bits/stdc++.h>
#define N 300001
using namespace std;
int T,a[105],dp[25005],n,ans=0;
int main()
{
cin>>T;
while(T--){
memset(dp,0,sizeof(dp));
cin>>n;
ans=n;
dp[0]=1;
for(int i=1;i<=n;i++)cin>>a[i];
sort(a+1,a+1+n);
for(int i=1;i<=n;i++){
if(dp[a[i]]){
ans--;
continue;
}
for(int j=a[i];j<=a[n];j++){
if(dp[j-a[i]])dp[j]=dp[j-a[i]];
}
}
cout<<ans<<endl;
}
}