洛谷P1658 购物
题目戳
题目描述
你就要去购物了,现在你手上有N种不同面值的硬币,每种硬币有无限多个。为了方便购物,你希望带尽量少的硬币,但要能组合出1到X之间的任意值。
输入输出格式
输入格式:
第一行两个数X、N,以下N个数,表示每种硬币的面值。
【数据规模】
对于30%的数据,满足N≤3,X≤20;
对于100%的数据,满足N≤10,X≤1000.
输出格式:
最少需要携带的硬币个数,如果无解输出-1.
输入输出样例
输入样例#1:
20 4
1 2 5 10
输出样例#1:
5
Solution:
一开始直接额考虑dp,以为是DAG上的DP,但是发现题意是1到x的所有值都要组成,而我开始的dp是固定终点的,So放弃。我们来考虑一下贪心:没有面额为1的直接输出-1,否则就取当前能取的最大值。什么意思呢?举个例子假设1、3、5,要组成10的面额,开始值为0,所以取1 -> 然后组成最大值为1,为了组成2,还是要取1 -> 最大值为2,此时大于等于面额3的硬币-1,所以选3 -> 此时最大值为5 ,大于等于5的面额-1 ->最大值变为了10,等于x,结束。。所以举的例子最少选4个硬币。为什么这样可行呢?因为小的面额必须先组成1到大面额的值-1的所有值(比如需要两个1来组成1、2),再去选大面额,这样大于大面额的值就以大面额为底加上小面额(比如以3为底,与两个1可以组成4、5)组成的最大值与下一个大面额的值-1比较,如果小于则再选一次刚刚的面额,直到大于等于为止便选当前的这个大面额,重复前面步骤,直到能组成的面额大于等于x停止。多举例子模拟理解,不难证明这样贪心是正确的且能保证硬币个数最小。
代码:
1 #include<bits/stdc++.h> 2 using namespace std; 3 #define ll long long 4 #define il inline 5 #define inf 233333333 6 int x,n,a[1005],sum,ans; 7 bool bj[100005]; 8 int main() 9 { 10 //freopen("shopping.in","r",stdin); 11 //freopen("shopping.out","w",stdout); 12 scanf("%d%d",&x,&n); 13 for(int i=1;i<=n;i++)scanf("%d",&a[i]),bj[a[i]]=1; 14 sort(a+1,a+n+1); 15 if(!bj[1])cout<<-1; 16 else { 17 while(sum<x) 18 { 19 for(int i=n;i>=1;i--) 20 if(sum+1>=a[i]){sum+=a[i],ans++;break;} 21 } 22 cout<<ans; 23 } 24 return 0; 25 }
PS:~蒟蒻写博客不易,转载请注明出处,万分感谢!~