Java教程

P5020 [NOIP2018 提高组] 货币系统

本文主要是介绍P5020 [NOIP2018 提高组] 货币系统,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

如果某种面额的货币可以被同样存在于货币系统中的一些面额更小的货币表示出,那么它删去与否等价。

当不能删时,从小到大考虑,因为不能出现面额更小的货币,所以该种面额的货币是必要的。

所以从小到大跑完全背包,每次判断是否必要即可。

尝试了一种二进制拆分(这里其实是倍增)加 bitset 优化完全背包的方法,这部分时间复杂度 \(O(\frac{na\log a}{w})\),对比普通完全背包的 \(O(na)\) 优势很大。

code:

#include<bits/stdc++.h>
using namespace std;
#define For(i,x,y)for(i=x;i<=(y);i++)
int a[105];
bitset<25005>f;
int read()
{
	int A;
	bool K;
	char C;
	C=A=K=0;
	while(C<'0'||C>'9')K|=C=='-',C=getchar();
	while(C>'/'&&C<':')A=(A<<3)+(A<<1)+(C^48),C=getchar();
	return(K?-A:A);
}
int main()
{
	int t,n,s,i,j;
	t=read();
	while(t--)
	{
		n=read();
		s=0;
		For(i,1,n)a[i]=read();
		sort(a+1,a+n+1);
		f=1;
		For(i,1,n)
		if(!f[a[i]])
		{
			j=a[i];
			while(j<=a[n])f|=f<<j,j<<=1;
			s++;
		}
		cout<<s<<endl;
	}
	return 0;
}
这篇关于P5020 [NOIP2018 提高组] 货币系统的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!