AcWing算法提高课-1.3.7货币系统

宣传一下算法提高课整理 <—

CSDN个人主页:更好的阅读体验 <—

f4e0159841ab450d861dde9e8fb5ba0d.gif

本题链接(AcWing)

点这里
题目描述

在网友的国度中共有  n n n 种不同面额的货币,第  i i i 种货币的面额为  a [ i ] a[i] a[i],你可以假设每一种货币都有无穷多张。

为了方便,我们把货币种数为  n n n、面额数组为  a [ 1.. n ] a[1..n] a[1..n] 的货币系统记作  ( n , a ) (n,a) (n,a)

在一个完善的货币系统中,每一个非负整数的金额  x x x 都应该可以被表示出,即对每一个非负整数  x x x,都存在  n n n 个非负整数  t [ i ] t[i] t[i] 满足  a [ i ] × t [ i ] a[i] × t[i] a[i]×t[i] 的和为  x x x

然而,在网友的国度中,货币系统可能是不完善的,即可能存在金额  x x x 不能被该货币系统表示出。

例如在货币系统  n = 3 ,   a = [ 2 , 5 , 9 ] n=3, a=[2,5,9] n=3,a=[2,5,9] 中,金额  1 , 3 1,3 1,3 就无法被表示出来。

两个货币系统  ( n , a ) (n,a) (n,a) 和  ( m , b ) (m,b) (m,b) 是等价的,当且仅当对于任意非负整数  x x x,它要么均可以被两个货币系统表出,要么不能被其中任何一个表出。

现在网友们打算简化一下货币系统。

他们希望找到一个货币系统  ( m , b ) (m,b) (m,b),满足  ( m , b ) (m,b) (m,b) 与原来的货币系统  ( n , a ) (n,a) (n,a) 等价,且  m m m 尽可能的小。

他们希望你来协助完成这个艰巨的任务:找到最小的  m m m

输入格式

输入文件的第一行包含一个整数  T T T,表示数据的组数。

接下来按照如下格式分别给出 T T T 组数据。

每组数据的第一行包含一个正整数  n n n

接下来一行包含  n n n 个由空格隔开的正整数  a [ i ] a[i] a[i]

输出格式

输出文件共有 T T T 行,对于每组数据,输出一行一个正整数,表示所有与  ( n , a ) (n,a) (n,a) 等价的货币系统  ( m , b ) (m,b) (m,b) 中,最小的  m m m

数据范围

1 ≤ n ≤ 100 1 \le n \le 100 1n100,
1 ≤ a [ i ] ≤ 25000 1 \le a[i] \le 25000 1a[i]25000,
1 ≤ T ≤ 20 1 \le T \le 20 1T20

输入样例:
2 
4 
3 19 10 6 
5 
11 29 13 19 17
输出样例:
2
5

思路

本题为DP问题,可以使用闫氏DP分析法解题。

DP:

当且仅当一张货币面额可以被该系统下其他货币表示时,此货币对该系统能表示的面额没有影响,换言之这个货币没有存在的必要,所以将这类的货币从系统中去除就可以得到等价的最小数量货币系统。

可以用 DP 求出能表示该面额的方案数,若对于一张货币方案数唯一(即只能被自己表示),则这张货币不能被省略,反之可以被省略。

  • 状态计算:
    ······ f [ 0 ] ← 1 f[0] \leftarrow 1 f[0]1
    ······ f [ j ] ← f [ j − a i ] f[j] \leftarrow f[j - a_i] f[j]f[jai]

因为有多组数据,所以要清空 f f f 数组


A C AC AC C o d e Code Code:

C + + C++ C++

#include <iostream>
#include <algorithm>
#include <cstring>

using namespace std; 

const int M = 25010, N = 110;

int n, T;
int f[M];
int a[N];

int main()
{
    scanf("%d", &T);
    
    while (T -- )
    {
        scanf("%d", &n);
    	for (int i = 0; i < n; i ++ )
            scanf("%d", &a[i]);
        
    	sort(a, a + n);
    	
    	int m = a[n - 1];
    	memset(f, 0, sizeof f);
    	f[0] = 1;
    	
    	int res = 0;
    	for (int i = 0; i < n; i ++ )
    	{
    		if (!f[a[i]]) res ++ ;
    		for (int j = a[i]; j <= m; j ++ )
    			f[j] += f[j - a[i]];
		}
		
		printf("%d\n", res);
	}
    
    return 0;
}

228aa7bed3e021faf24cf8560d3e47bb.gif

最后,如果觉得对您有帮助的话,点个赞再走吧!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

星河依旧长明

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

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

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

打赏作者

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

抵扣说明:

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

余额充值