首页 > 代码库 > [题解]noip邮票面值设计

[题解]noip邮票面值设计

描述

给定一个信封,最多只允许粘贴N张邮票,计算在给定M(N+M<=10)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大max ,使得1~max之间的每一个邮资值都能得到。

例如,N=3,M=2,如果面值分别为1分、4分,则在l分~6分之间的每一个邮资值都能得到(当然还有8分、9分和12分):如果面值分别为1分、3分,则在1分~7分之间的每一个邮资值都能得到。可以验证当N=3,M=2时,7分就是可以得到连续的邮资最大值,所以MAX=7,面值分别为l分、3分。

样例输入:共一行,两个整数,分表为N与M的值。

格式

输入格式

一行,分别为N,M。

输出格式

两行。

第一行为m种邮票的面值,按升序排列,各数之间用一个空格隔开。

第二行为最大值。

如果有多解,输出字典序最大的一个。

样例1

样例输入1[复制]

 
3 2

样例输出1[复制]

 
1 3MAX=7

限制

各个测试点1s

来源

NOIP1999

(转自https://vijos.org/p/1179)


  这道题首先看数据范围就大概能够猜出是搜索,可是,出现了以下问题

  1)搜索的范围,不可能从1搜到1000

  2)如何求MAX

  先来思考问题二,首先回顾一下求出面值为i所需最少的邮票怎么求,用动规对不对?

f[i] = min(f[i - 面额] + 1, f[i]),这样求的对不对?

  那从1一直往下做,求到某个f[i] > n,说明什么?i这个面值不能用要超过n张的邮票

凑出来,故MAX = i - 1

  问题二解决了,然后来解决问题一,首先上限,很容易想到,当前的MAX + 1,为什么

不能是MAX + 2,或者更多,因为MAX + 1就凑不出来了。然后来思考下限,前一个面额 + 1?

如果比它少会发生什么?MAX貌似不会改变。所以得到了这个范围[前一个面额 + 1, 当前MAX]

  接下来附上代码


 

Code

 1 /** 2  * vijos.org 3  * Problem#1179 4  * Accepted 5  * Time:107ms 6  * Memory:568k 7  */ 8 #include<iostream> 9 #include<cstring>10 #include<cstdio>11 #include<cstdlib>12 using namespace std;13 int buf[45];14 int ans[45];15 int f[1001];16 int n ,k;17 int maxv = 0;18 int cale(int count){19     if(!count)    return 0;20     memset(f, 0x7f, sizeof(f));21     f[0] = 0;22     int i = 1;23     do{24         for(int j = 1;j <= count && buf[j] <= i;j++)25             f[i] = min(f[i - buf[j]] + 1, f[i]);26     }while(f[i++] <= n);27     return i - 2;28 }29 void search(int sec){30     if(sec >= k){31         int c = cale(k);32         if(c <= maxv)    return;33         for(int i = 1;i <= k; i++)34             ans[i] = buf[i];35         maxv = c;36         return;37     }else{38         int temp = cale(sec);39         for(int i = temp + 1;i > buf[sec]; i--){40             buf[sec + 1] = i;41             search(sec + 1);42         }43     }44 }45 int main(){46     cin >> n >> k;47     search(0);48     for(int i = 1;i <= k;i++)49         cout<<ans[i]<<" ";50     cout<<endl;51     cout<<"MAX="<<maxv;52     return 0;53 }

 

[题解]noip邮票面值设计