[NOIP2018 提高组] 货币系统

题目描述

在网友的国度中共有 𝑛 种不同面额的货币,第 𝑖种货币的面额为 𝑎[𝑖],你可以假设每一种货币都有无穷多张。为了方便,我们把货币种数为 𝑛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;
	}
}

  • 20
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值