题目链接:https://vjudge.net/contest/103424#problem/K

转载于:https://blog.csdn.net/acm_davidcn/article/details/5549933

题目大意:

给n个物品,和m块钱,输出能够购买最多物品的个数和购买这么多物品的方案数。

 解题分析:

背包的一种进化版 , 除了记录最多能买多少个 , 需要记录买这么多个的方法 , 所以要在二维的基础上加多一维 .

状态转移方程如下 : f[i][j][k]=f[i-1][j-t[i]][k-1]+f[i-1][j][k];

( 前 i 个物品在有 j 元的时候买 k 个物品的方法 ,t[i] 为第 i 个物品的价格 )

#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std;
int val[], dp[][][]; //dp[i][j][k]表示有k块钱,在前i种物品买j个物品的数量
int main() {
int i, j, k, t, n, m, sign;
scanf("%d", &t);
while (t--) {
scanf("%d%d", &n, &m);
for (i = ; i <= n; i++)
scanf("%d", &val[i]);
memset(dp, , sizeof(dp));
for (i = ; i <= n; i++)
for (k = ; k <= m; k++)
dp[i][][k] = ; //初始化,一样不买种类是1
for (i = ; i <= n; i++) {
for (j = ; j <= i; j++)
for (k = m; k >= ; k--) {
if (k >= val[i])
dp[i][j][k] += dp[i - ][j - ][k - val[i]] + dp[i - ][j][k]; //加上买和不买的情况
}
}
sign = ;
for (i = n; i >= ; i--) {
if (dp[n][i][m] != ) { //找购买方案数不为0的最大购买数量i
printf("You have %d selection(s) to buy with %d kind(s) of souvenirs.\n", dp[n][i][m], i);
sign = ;
break;
}
}
if (!sign)
puts("Sorry, you can't buy anything.");
}
return ;
}

第二种方法,对上述代码进行优化

dp[i][j][0]表示前i个物品有j元时的最多物品数,dp[i][j][1]用来储存方案数

***设当前所选的物品为i
1.  若选了物品i后,能买的件数比不选物品i的件数大,即dp[j - val[i]]>dp[j]
那么更新dp[j],同时,dp[j][1]的方案数即为dp[j - val[i]][1]
原因是:假设dp[j - val[i]]的方案数为 AB AC 两种,那么在此情况下加个D,为ABD, ACD,仍为两种,所以dp[j] = f[j - val[i]]即可
当然,要注意dp[j - val[i]]为0的情况,因此当它为0时,dp[j] = 1,1即为D

2.  若选了物品i后,能买的件数比不选物品i的件数相同,即dp[j - val[i]] == dp[j]
即原先不选第i个物品,所需要的方案数为dp[j];而选了物品i的方案数为dp[j - val[i]]。
因此,总的方案数即为dp[j] + dp[j - val[i]]
当然,这里也要注意dp[j - val[i]] = 0的情况,当它为0时,dp[j] += 1,1即为D

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define mem(p,k) memset(p,k,sizeof(p))
using namespace std;
int p[], dp[][]; //dp[i][j][0]表示前i个物品,最多有j元,能买多少种物品
int main() { //dp[i][j][1]表示前i个物品,最多有j元,能买最多物品的方案数
int t, cur = ;
cin >> t;
while (t--) {
int m, n;
scanf("%d%d", &n, &m);
for (int i = ; i <= n; i++)scanf("%d", p + i);
mem(dp, );
for (int i = ; i <= m; i++)dp[i][] = ; //初始化,钱数为j,一个物品都不买的方案数为1(因为此时对应的dp[i][0]=0,表示一个物品都没买)
for (int i = ; i <= n; i++) {
for (int j = m; j >= p[i]; j--) {
if (dp[j][] == dp[j - p[i]][] + ) { //若选了物品i后,能买的件数比不选物品i的件数相同
if (!dp[j - p[i]][])dp[j][] += ;
else
dp[j][] += dp[j - p[i]][];
}
else if(dp[j][]<dp[j - p[i]][] + ) { //若选了物品i后,能买的件数比不选物品i的件数大
dp[j][] = dp[j - p[i]][] + ;
if (!dp[j - p[i]][])dp[j][] = ;
else
dp[j][] = dp[j - p[i]][];
}
}
}
if (dp[m][]) {
printf("You have %d selection(s) to buy with %d kind(s) of souvenirs.\n", dp[m][], dp[m][]);
}
else printf("Sorry, you can't buy anything.\n");
}
}

2018-05-21

hdu 2126 Buy the souvenirs 【输出方案数】【01背包】(经典)的更多相关文章

  1. hdu 2126 Buy the souvenirs 二维01背包方案总数

    Buy the souvenirs Time Limit: 10000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

  2. HDU 2126 Buy the souvenirs (01背包,输出方案数)

    题意:给出t组数据 每组数据给出n和m,n代表商品个数,m代表你所拥有的钱,然后给出n个商品的价值 问你所能买到的最大件数,和对应的方案数.思路: 如果将物品的价格看做容量,将它的件数1看做价值的话, ...

  3. hdu 2126 Buy the souvenirs(记录总方案数的01背包)

    Buy the souvenirs Time Limit: 10000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

  4. hdu 2126 Buy the souvenirs 买纪念品(01背包,略变形)

    题意: 给出一些纪念品的价格,先算出手上的钱最多能买多少种东西k,然后求手上的钱能买k种东西的方案数.也就是你想要买最多种东西,而最多种又有多少种组合可选择. 思路: 01背包.显然要先算出手上的钱m ...

  5. [HDU 2126] Buy the souvenirs (动态规划)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2126 题意:给你n个物品,m元钱,问你最多能买个多少物品,并且有多少种解决方案. 一开始想到的是,先解 ...

  6. hdu 3339 In Action(迪杰斯特拉+01背包)

    In Action Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

  7. HDU 1203 I NEED A OFFER!(01 背包DP)

    点我看题目 题意 : 中文题不详述. 思路 :类似于01背包的DP,就是放与不放的问题,不过这个要求概率,至少得到一份offer的反面就是一份也得不到,所以先求一份也得不到的概率,用1减掉就可以得到所 ...

  8. HDU 2639 骨头收集者 II【01背包 】+【第K优决策】

    题目链接:https://vjudge.net/contest/103424#problem/H 题目大意:与01背包模板题类似,只不过要我们求第K个最大的总价值. 解题分析: 其基本思想是将每个状态 ...

  9. HDU 5887 Herbs Gathering(搜索求01背包)

    http://acm.hdu.edu.cn/showproblem.php?pid=5887 题意: 容量很大的01背包. 思路: 因为这道题目背包容量比较大,所以用dp是行不通的.所以得用搜索来做, ...

随机推荐

  1. POJ1679 The Unique MST【次小生成树】

    题意: 判断最小生成树是否唯一. 思路: 首先求出最小生成树,记录现在这个最小生成树上所有的边,然后通过取消其中一条边,找到这两点上其他的边形成一棵新的生成树,求其权值,通过枚举所有可能,通过这些权值 ...

  2. python - 用类写装饰器

    这里用到了__call__的class内置参数 #类装饰器: class zsq(): #本质是定义一个参数,让装饰的主题传递至__call__方法内部 def __init__(self,obj): ...

  3. C# 无法识别的消息版本。

    问题:最近跟OA的java项目做审核接口,调用接口时提示"无法识别的消息版本.“ 解决:一直以为是协议不兼容,查了半天,最终发现是这个项目.net framework版本太低,升级为高版本即 ...

  4. 【漏洞挖掘】攻击对外开放的Docker API接口

    https://medium.com/@riccardo.ancarani94/attacking-docker-exposed-api-3e01ffc3c124 1)场景 攻击开放在互联网的Dock ...

  5. Linux下rsyslog日志收集服务环境部署记录【转】

    rsyslog 可以理解为多线程增强版的syslog. 在syslog的基础上扩展了很多其他功能,如数据库支持(MySQL.PostgreSQL.Oracle等).日志内容筛选.定义日志格式模板等.目 ...

  6. 常用的4个eclipse插件安装过程及使用方法

    最近整合了4个常用eclipse插件安装过程,分别是PMD.checkstyle.findbugs.sourcemonitor插件.因为我这里没有外网,所以所有的插件不是最新版,建议有网的童鞋自行在外 ...

  7. SKlearn库学习曲线

    思想: # 1.现将所有样本用交叉验证方法或者(随机抽样方法) 得到 K对 训练集-验证集# 2.依次对K个训练集,拿出数量不断增加的子集如m个,并在这些K*m个子集上训练模型.# 3.依次在对应训练 ...

  8. PYTHON-组合 封装 多态 property装饰器

    # 组合'''软件重用的重要方式除了继承之外还有另外一种方式,即:组合组合指的是,在一个类中以另外一个类的对象作为数据属性,称为类的组合 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之 ...

  9. webstrom随手笔记

    1.node代码智能提示功能: 2.页面代码格式化快捷键: Ctrl+Alt+L键,或者Code =>Rearrange code

  10. 【svn】svn的使用

    直接看这个教程:http://www.runoob.com/svn/svn-tutorial.html 里面讲得很有条理,很容易看懂.提供了指令以及相应的名词解释.不像其他网上的资料,干巴巴的只有一对 ...