zoukankan      html  css  js  c++  java
  • 洛谷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 }
  • 相关阅读:
    base64模块的使用
    14-类的结构之一:属性
    13-类和对象
    12-数组的常见异常
    11-Arrays工具类的使用
    10-二维数组
    09-一维数组
    08-数组的概述
    07-流程控制
    06-运算符
  • 原文地址:https://www.cnblogs.com/five20/p/7804757.html
Copyright © 2011-2022 走看看