UVa 242 Stamps and Envelope Size (无限背包,DP)

题意:信封上最多贴S张邮票。有N个邮票集合,每个集合有不同的面值。问哪个集合的最大连续邮资最 大,输出最大连续邮资和集合元素。

最大连续邮资是用S张以内邮票面值凑1,2,3...到n+1凑不出来了,最大连续邮资就是n。如果不止一个集合结果相 同,输出集合元素少的,

如果仍相同,输出最大面值小的。

析:这个题,紫书上写的不全,而且错了好几次,结果WA好几次。

首先这个和背包问题差不多,我们只用一维就好。dp[i]表示邮资为 i 时的最小邮票数,然后,如果dp[i] > s就该结束了。

其他的就很简单了,主要是我没理解题意。

代码如下:

#pragma comment(linker, "/STACK:1024000000,1024000000")
#include <cstdio>
#include <string>
#include <cstdlib>
#include <cmath>
#include <iostream>
#include <cstring>
#include <set>
#include <queue>
#include <algorithm>
#include <vector>
#include <map>
#include <cctype>
#include <cmath>
#include <stack>
#define freopenr freopen("in.txt", "r", stdin)
#define freopenw freopen("out.txt", "w", stdout)
using namespace std;

typedef long long LL;
typedef pair<int, int> P;
const int INF = 0x3f3f3f3f;
const double inf = 0x3f3f3f3f3f3f;
const LL LNF = 0x3f3f3f3f3f3f;
const double PI = acos(-1.0);
const double eps = 1e-8;
const int maxn = 15 + 5;
const int mod = 1e9 + 7;
const int dr[] = {-1, 0, 1, 0};
const int dc[] = {0, 1, 0, -1};
const char *de[] = {"0000", "0001", "0010", "0011", "0100", "0101", "0110", "0111", "1000", "1001", "1010", "1011", "1100", "1101", "1110", "1111"};
int n, m;
const int mon[] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
const int monn[] = {0, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
inline int Min(int a, int b){ return a < b ? a : b; }
inline int Max(int a, int b){ return a > b ? a : b; }
inline LL Min(LL a, LL b){ return a < b ? a : b; }
inline LL Max(LL a, LL b){ return a > b ? a : b; }
inline bool is_in(int r, int c){
    return r >= 0 && r < n && c >= 0 && c < m;
}
int a[maxn][maxn];
int dp[1024];
int ans[maxn];

int main(){
    while(scanf("%d", &n) == 1 && n){
        scanf("%d", &m);
        for(int i = 0; i < m; ++i){
            scanf("%d", &a[i][0]);
            for(int j = 1; j <= a[i][0]; ++j)
                scanf("%d", &a[i][j]);
            fill(dp, dp+1024, INF);
            dp[0] = 0;
            for(int j = 1; j < 1024; ++j){
                for(int k = 1; k <= a[i][0] && j-a[i][k] >= 0; ++k){
                    dp[j] = Min(dp[j], dp[j-a[i][k]]+1);
                }
                if(dp[j] > n){  ans[i] = j-1;  break; }
            }
        }

        int anss = -1, cnt = 0;
        for(int i = 0; i < m; ++i){
            if(ans[i] > anss){  anss = ans[i];  cnt = i; }
            else if(ans[i] == anss && a[i][0] < a[cnt][0])  cnt = i;
            else if(ans[i] == anss && a[i][0] == a[cnt][0]){
                bool ok = false;
                for(int j = a[i][0]; j >= 0; --j)
                    if(a[i][j] < a[cnt][j]){ ok = true;  break; }
                    else if(a[i][j] > a[cnt][j])  break;
                if(ok)  cnt = i;
            }
        }
        printf("max coverage =%4d :", anss);
        for(int i = 1; i <= a[cnt][0]; ++i)  printf("%3d", a[cnt][i]);
        printf("\n");
    }
    return 0;
}
时间: 2024-12-21 07:13:58

UVa 242 Stamps and Envelope Size (无限背包,DP)的相关文章

UVa 242 - Stamps and Envelope Size(DP)

给出一个s,然后给出n组邮票,问那一组可以凑出最大连续邮资. 对每一组邮票,求出当邮资为i时需要邮票数的最小值d[i],边界为d[0]=0.d[i]>s时break.类似于背包问题的求法,具体方法见代码. #include<cstdio> #include<cstring> #include<algorithm> using namespace std; const int maxn=1010; int d[maxn],ans[20],num[20],a[20][

UVa 242 邮票和信封(完全背包)

https://vjudge.net/problem/UVA-242 题意: 输入s(每个信封能粘贴的最多邮票数量)和若干邮票组合,选出最大连续邮资最大的一个组合(最大连续邮资也就是用s张以内的邮票来凑1,2,3,4...n,如果无法凑成n+1,那么最大值也就是n了).如果有多个最大值,则优先考虑邮票数少的,其次考虑邮票面值最大的那个更小的. 思路: 完全背包问题. 完全背包是物品无限,在这里和题意相符合,每种邮票也是可以无限使用的.最大连续邮资就相当于一个背包容量,d[i]表示当最大连续邮资为

uva242,Stamps and Envelope Size

这题紫薯上翻译错了 应该是:如果有多个并列,输出邮票种类最少的那个,如果还有并列,输出最大面值最小的那个 坑了我一个下午 dp[p][q]==1表示可以用不超过q张组成面额p 结合记忆化,p从1开始枚举,一直枚举找到dp[p][q]=0的时候就可以了 这题应该归类成一种背包吧 注意dp初始化的时候应该初始化为-1(我就因为粗心,tle好久) 最后输出的时候比较恶心 最终的修改后的代码 实验证明,先读入所有数据后再处理比边读数据边处理要快 /* * Author: Bingo * Created

uva 242

242 - Stamps and Envelope Size Time limit: 3.000 seconds  Stamps and Envelope Size  Philatelists have collected stamps since long before postal workers were disgruntled. An excess of stamps may be bad news to a country's postal service, but good news

UVA - 10280Old Wine Into New Bottles(完全背包+剪枝)

题目:UVA - 10280Old Wine Into New Bottles(完全背包+剪枝) 题目大意:现在要将旧酒装入新瓶中,每种瓶子都有最小最大的容量要求,然后给你L升酒,在给你N个瓶子,每中瓶子的规格说明也给你,每个种类的瓶子的供应是无限的,问怎样子安排这些酒才能使得剩余的酒最少. 解题思路:这题是完全背包的题目,但是一开始就被这题的数据吓到,10^9ML,然后还有100个瓶子,瓶子的最大最小容量还相差4000左右,直接去完全背包肯定超时.之后看了大神的题接,有个规律:每个种类的瓶子的

uva 165 Stamps (回溯)

uva 165 Stamps The government of Nova Mareterrania requires that various legal documents have stamps attached to them so that the government can derive revenue from them. In terms of recent legislation, each class of document is limited in the number

RQNOJ 671 纯洁的买卖:无限背包

题目链接:https://www.rqnoj.cn/problem/671 题意: ALEJ要通过倒卖东西来赚钱. 现在他有m元经费. 有n种物品供他选择,每种物品数量无限. 第i件物品的买入价为c[i],卖出价为r[i],每卖出一件物品i后,要交c[i]的税. 问:一次买卖之后,经费最多有多少. 题解: 注:(1)"买"和"卖"是有顺序的. 也就是说,收购一件物品所得到的"未来利润"并不能当作现在的经费来用. (2)"缴税"

01背包与物品无限背包

一.01背包问题 01背包是在M件物品取出若干件放在空间为W的背包里,每件物品的体积为C1,C2,…,Cn,与之相对应的价值为W1,W2,…,Wn.求解将那些物品装入背包可使总价值最大. 动态规划: 1) 子问题定义:F[i][j]表示前i件物品中选取若干件物品放入剩余空间为j的背包中所能得到的最大价值. 2) 根据第i件物品放或不放进行决策 其中F[i-1][j]表示前i-1件物品中选取若干件物品放入剩余空间为j的背包中所能得到的最大价值: 而F[i-1][j-C[i]]+W[i]表示前i-1

projecteuler----&gt;problem=31----Coin sums 无限背包计算可能存在的次数

Problem 31 In England the currency is made up of pound, £, and pence, p, and there are eight coins in general circulation: 1p, 2p, 5p, 10p, 20p, 50p, £1 (100p) and £2 (200p). It is possible to make £2 in the following way: 1×£1 + 1×50p + 2×20p + 1×5p