zoukankan      html  css  js  c++  java
  • POJ 3260 The Fewest Coins(背包问题)

    【题目链接】 http://poj.org/problem?id=3260

    【题目大意】

      给出你拥有的货币种类和每种的数量,商店拥有的货币数量是无限的,
      问你买一个价值为m的物品,最少的货币流通数量为多少

    【题解】

      我们可以计算出买不同价值的物品,在没有找钱情况下的最少用币数量,
      记为dp[i],这个可以用多重背包来完成,对于商店来说,我们可以计算出,
      对于不同额度的找零,最少用币数量,记为f[i],
      我们发现,对于货币流通最少,就是求dp[i+m]+f[i]的最小值,
      由鸽巢原理可得我们只要计算maxw*maxw+m的dp值就可以了。

    【代码】

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    const int INF=0x3f3f3f3f;
    int m,n,w[110],c[110],dp[24563],f[24563]; 
    int main(){
        while(~scanf("%d%d",&n,&m)){
            int maxn=0;
            for(int i=1;i<=n;i++)scanf("%d",&w[i]),maxn=max(maxn,w[i]);
            for(int i=1;i<=n;i++)scanf("%d",&c[i]);
            maxn=maxn*maxn+m+1;
            memset(dp,INF,sizeof(dp));
            dp[0]=0;
            for(int i=1;i<=n;i++){
                if(c[i]*w[i]>=maxn){
                    for(int j=w[i];j<=maxn;j++)dp[j]=min(dp[j],dp[j-w[i]]+1);
                }else{
                    for(int k=1;k<c[i];k<<=1){
                        for(int j=maxn;j>=w[i]*k;j--)dp[j]=min(dp[j],dp[j-w[i]*k]+k);
                        c[i]-=k;
                    }for(int j=maxn;j>=w[i]*c[i];j--)dp[j]=min(dp[j],dp[j-w[i]*c[i]]+c[i]);
                }memset(f,INF,sizeof(f));f[0]=0;
            }
            for(int i=1;i<=n;i++){
                for(int j=w[i];j<=maxn;j++)f[j]=min(f[j],f[j-w[i]]+1);
            }int ans=INF;
            for(int i=0;i<=maxn-m;i++)ans=min(ans,dp[i+m]+f[i]);
            if(ans==INF)ans=-1;
            printf("%d
    ",ans);
        }return 0;
    }
  • 相关阅读:
    mysql中的enum型
    mysql中的时间year/date/time/datetime
    一些数字的属性
    mysql增删
    Perl6 Bailador框架(8):自定义400/500
    react: typescript jest && enzyme
    webstorm tslint配置
    react: typescript import images alias
    JSONP原理及简单实现
    纯css画三角形
  • 原文地址:https://www.cnblogs.com/forever97/p/poj3260.html
Copyright © 2011-2022 走看看