【题意】:已知整个交易系统有N
(1 ≤ N ≤ 100)种不同的货币,分别价值V1,V2,V3.......VN(1 ≤ Vi ≤ 120),FJ分别有C1,C2,C3.....CN(0
≤ Ci ≤10,000)张相应价值货币。FJ只能用仅有的货币去买价值T(1 ≤T≤10,000)分的东西,而老板有无数的货币可以找给他。求FJ给老板的货币数+老板找给FJ的货币数的最小值。

Input

* Line 1: Two space-separated integers: N and T.
* Line 2: N space-separated integers, respectively V1, V2, ..., VN coins (V1...VN).
* Line 3: N space-separated integers, respectively C1, C2, ..., CN

Output

* Line 1: A line containing a single integer, the minimum number of coins involved in a payment and
change-making. If it is impossible for Farmer John to pay and receive exact change, output -1.

Sample Input

3 70
5 25 50
5 2 1

Sample Output

3

【分析】:将问题拆分为两部分:(多重背包+完全背包)FJ支付的货币数实是多重背包问题,而老板找给FJ的货币数可以转化为完全背包问题。

思路一:FJ最多可以给老板(N+他所拥有的货币最大面值),用t[i]保存在此区间FJ支付给老板的钱数为i时所需的最小货币数。对于[N,N+Max]区间,用l[j]表示老板找给FJ的钱数为j时的最小货币数(其中i-j=N),最终求出所有t[i]+l[j]的最小值,即为题目所求。

<span style="font-size:14px;">#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
int v[110],c[110],t[24400],l[24400];
int T=0,N=0;
int Max;
const int inf =10000000;
void  ZeroOnePack(int vi,int k)
{
     for(int i=N+Max;i>=vi;i--)
            t[i]=min(t[i],t[i-vi]+k);
}
void CompletePack(int vi,int* t,int n)
{
    for(int i=vi;i<=n;i++)
        t[i]=min(t[i],t[i-vi]+1);
}
void MultiplePack(int vi,int ci)
{
    if(vi*ci>N+Max)
          CompletePack(vi,t,N+Max);
    int k=1;
    while(k<ci)
    {
        ZeroOnePack(k*vi,k);
        ci=ci-k;
        k=2*k;
    }
    ZeroOnePack(ci*vi,ci);
}
int main (void)
{
    while(~scanf("%d%d",&T,&N))
    {
         int Min=inf;
         Max=0;
         for(int i=1;i<24400i++) t[i]=inf;
         t[0]=0;
         for(int i=0;i<T;i++)
        {
            scanf("%d",&v[i]);
            if(Max<v[i])    Max=v[i];
        }
         Max=Max*Max;//保证背包足够大
      
        for(int i=0;i<T;i++)  scanf("%d",&c[i]);
        for(int i=0;i<T;i++) MultiplePack(v[i],c[i]);
        for(int i=N;i<=N+Max;i++)
        {
            if(t[i]!=inf)
            {
                for(int i=0;i<24400;i++) l[i]=inf;
                l[0]=0;
                for(int j=0;j<T;j++)
                    CompletePack(v[j],l,i-N);
                Min=min(Min,t[i]+l[i-N]);
            }
        }
        if( Min==inf)   Min=-1;
        printf("%d\n",Min);
    }
} </span>

【Spotted】

1.最初不知道应该开多大的数组保存货币数,在别人的博客中找到以下内容:

最重要的是上界的处理。可以注意到,上界为maxw*maxw+m(maxw最大面额的纸币),也就是24400元。(网上的证明)证明如下:
如果买家的付款数大于了maxw*maxw+m,即付硬币的数目大于了maxw,根据鸽笼原理,至少有两个的和对maxw取模的值相等,也就是说,这部分硬币能够用更少的maxw来代替。证毕。

【【离散不是白学的呀呵呵!

2.memset和fill相关。。。。在下一篇博客中。。。。

思路二:使用同一个数组【想了好久都感觉无解,看了别人的代码才顿悟。。。。窒息。。。。】

改一下main函数,代码如下:

<span style="font-family:Microsoft YaHei;font-size:14px;">int main (void)
{
while(~scanf("%d%d",&T,&N))
{
int Min=inf;
Max=0;
for(int i=1;i<24400;i++) t[i]=inf;
t[0]=0;
for(int i=0;i<T;i++)
{
scanf("%d",&v[i]);
if(Max<v[i]) Max=v[i];
}
Max=Max*Max;//保证背包足够大 for(int i=0;i<T;i++) scanf("%d",&c[i]);
for(int i=0;i<T;i++) MultiplePack(v[i],c[i]);
for(int i=0;i<T;i++)
{
for(int k= N+Max-v[i]; k>=N; k--)
t[k]=min(t[k],t[k+v[i]]+1);
//由于是老板还给FJ的钱,且是完全背包问题,所以正负号相反,且逆序进行循环
}
if( t[N]==inf) t[N]=-1;;
printf("%d\n",t[N]);
}
}</span>

SOJ 2749_The Fewest Coins的更多相关文章

  1. POJ3260The Fewest Coins[背包]

    The Fewest Coins Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 6299   Accepted: 1922 ...

  2. POJ3260——The Fewest Coins(多重背包+完全背包)

    The Fewest Coins DescriptionFarmer John has gone to town to buy some farm supplies. Being a very eff ...

  3. The Fewest Coins POJ - 3260

    The Fewest Coins POJ - 3260 完全背包+多重背包.基本思路是先通过背包分开求出"付出"指定数量钱和"找"指定数量钱时用的硬币数量最小值 ...

  4. POJ 3260 The Fewest Coins(多重背包+全然背包)

    POJ 3260 The Fewest Coins(多重背包+全然背包) http://poj.org/problem?id=3260 题意: John要去买价值为m的商品. 如今的货币系统有n种货币 ...

  5. POJ 3260 The Fewest Coins(完全背包+多重背包=混合背包)

    题目代号:POJ 3260 题目链接:http://poj.org/problem?id=3260 The Fewest Coins Time Limit: 2000MS Memory Limit: ...

  6. (混合背包 多重背包+完全背包)The Fewest Coins (poj 3260)

    http://poj.org/problem?id=3260   Description Farmer John has gone to town to buy some farm supplies. ...

  7. POJ3260:The Fewest Coins(混合背包)

    Description Farmer John has gone to town to buy some farm supplies. Being a very efficient man, he a ...

  8. POJ 3260 The Fewest Coins(多重背包问题, 找零问题, 二次DP)

    Q: 既是多重背包, 还是找零问题, 怎么处理? A: 题意理解有误, 店主支付的硬币没有限制, 不占额度, 所以此题不比 1252 难多少 Description Farmer John has g ...

  9. 洛谷P2851 [USACO06DEC]最少的硬币The Fewest Coins(完全背包+多重背包)

    题目描述 Farmer John has gone to town to buy some farm supplies. Being a very efficient man, he always p ...

随机推荐

  1. [Tunny]Grunt基础介绍

    [黄映焜/Tunny,20140711] Grunt是一个JavaScript任务管理器,对于需要反复重复的任务,例如压缩.编译.单元测试.代码检查等,自动化工具可以减轻你的劳动,简化你的工作. 本文 ...

  2. netstat -a 显示出你的计算机当前所开放的所有端口

    netstat -s -e 比较详细的显示你的网络资料,包括TCP.UDP.ICMP 和 IP的统计等 

  3. CAD绘制固定矩形标注(网页版)

    js中实现代码说明: function DoFixRectComment() { var ent = mxOcx.DrawCustomEntity("TestMxCustomEntity&q ...

  4. 阿里云部署flask

    https://www.cnblogs.com/Ray-liang/p/4173923.html

  5. Spring boot 配置tomcat后 控制台不打印SQL日志

    在pom.xml中配置tomcat启动处加上: <dependency> <groupId>org.springframework.boot</groupId> & ...

  6. promise的简单使用

    var p = new Promise(function (resolve,reject) { /*setTimeout(function () { resolve('success') },3000 ...

  7. ''tensorflow.python.framework.errors_impl.ResourceExhaustedError: OOM when allocating tensor with shape[?]'' 错误分析

    这是tensorflow 一个经常性错误,错误的原因在于:显卡内存不够. 解决方法就是降低显卡的使用内存,途径有以下几种措施: 1 减少Batch 的大小 2 分析错误的位置,在哪一层出现显卡不够,比 ...

  8. NOIP专题复习3 图论-强连通分量

    目录 一.知识概述 二.典型例题 1.[HAOI2006]受欢迎的牛 2.校园网络[[USACO]Network of Schools加强版] 三.算法分析 (一)Tarjan算法 (二)解决问题 四 ...

  9. jenkins构建项目记录2(tag)

    与jenkins构建项目记录1不同的是通过tag拉去对应版本代码 1.先安装创建(git parameter) 2.general设置 name可任意命名,下面源码管理设置时变量会引用到. 3.源码管 ...

  10. [Python3网络爬虫开发实战] 1.8.3-Scrapy-Splash的安装

    Scrapy-Splash是一个Scrapy中支持JavaScript渲染的工具,本节来介绍它的安装方式. Scrapy-Splash的安装分为两部分.一个是Splash服务的安装,具体是通过Dock ...