☆【UVA - 624 】CD(dp + 0-1背包 + 记录路径)

题干:

You have a long drive by car ahead. You have a tape recorder, but unfortunately your best music is on CDs. You need to have it on tapes so the problem to solve is: you have a tape N minutes long. How to choose tracks from CD to get most out of tape space and have as short unused space as possible.


Assumptions:

  • number of tracks on the CD. does not exceed 20
  • no track is longer than N minutes
  • tracks do not repeat
  • length of each track is expressed as an integer number
  • N is also integer

Program should find the set of tracks which fills the tape best and print it in the same sequence as the tracks are stored on the CD

Input

Any number of lines. Each one contains value N, (after space) number of tracks and durations of the tracks. For example from first line in sample data: N=5, number of tracks=3, first track lasts for 1 minute, second one 3 minutes, next one 4 minutes

Output

Set of tracks (and durations) which are the correct solutions and string ``sum:" and sum of duration times.

Sample Input

5 3 1 3 4
10 4 9 8 4 2
20 4 10 5 7 4
90 8 10 23 1 2 3 4 5 7
45 8 4 10 44 43 12 9 8 2

Sample Output

1 4 sum:5
8 2 sum:10
10 5 4 sum:19
10 23 1 2 3 4 5 7 sum:55
4 10 12 9 8 2 sum:45

 

解题报告:

这题本来想用排序去做(因为可以记录路径),但是发现排序了反而麻烦,而直接这样做就非常的巧妙。这样pre用一维数组相当于实时更新,所以最后输出的路径也是正确的,倒序读入反而就是错误的了。

AC代码:

#include<bits/stdc++.h>using namespace std;
int dp[100000 + 5],pre[100000 + 5];
int n,m;
struct Node {int id;int val;
} node[25];
bool cmp(Node a,Node b) {return a.val > b.val;
}
void print(int loc) {if(pre[loc] == 0) {printf("%d ",loc);return;}print(pre[loc]);printf("%d ",loc - pre[loc]);}
int main()
{while(~scanf("%d%d",&m,&n)) {memset(dp,0,sizeof(dp));for(int i = 1; i<=n; i++) {scanf("%d",&node[i].val);node[i].id = i;}
//		sort(node+1,node+n+1,cmp);for(int i = 1; i<=n; i++) {for(int j = m; j>=node[i].val; j--) {
//				if(dp[j] !=0) continue;
//				dp[j] = max(dp[j],dp[j-node[i].val ] + node[i].val);
//				poe[j] = j-node[i].val;if(dp[j-node[i].val ] + node[i].val > dp[j]) {dp[j] = dp[j-node[i].val ] + node[i].val;pre[j] = j - node[i].val;}}}print(dp[m]);printf("sum:%d\n",dp[m]);}return 0 ;} 

错误代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define maxn 100005
using namespace std;
//  45 3 38 39 2
int dp[maxn];
int num[25];
int vis[25][maxn];
int n,m;
int main()
{while(scanf("%d%d",&n,&m)!=EOF){memset(dp,0,sizeof(dp));memset(vis,0,sizeof(vis));memset(num,0,sizeof(num));for(int i=0;i<m;i++)scanf("%d",&num[i]);for(int i=0;i<=m-1;i++)//这里正序是错的!!倒序是对的!用上面给的那个样例试一下就知道了。sum输出的是对的但是路径是错的。 for(int j=n;j>=num[i];j--){if(dp[j-num[i]]+num[i]>dp[j]){dp[j]=dp[j-num[i]]+num[i];vis[i][j]=1;}}
//		printf("vis[1]][n] = %d\n",vis[0][n]);int j=n;for(int i=0;i<m;i++){if(vis[i][j]){printf("%d ",num[i]);j-=num[i];}} printf("sum:%d\n",dp[n]);}return 0;
}

这样做0-1背包的话sum的值是正确的但是输出的路径是错误的,因为你这里的vis不带更新(因为是二维的),所以还是旧值,要避免这个问题就是倒序做0-1背包。

(ps:其实这个题改动一下还是可以正序跑背包的,就是输出的时候倒序处理一下就行了,然后用栈存起来最后顺序输出就可以了  下面贴一下代码吧。。)

AC代码2:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define maxn 100005
using namespace std;
//  45 3 38 39 2
int dp[maxn];
int num[25];
int vis[25][maxn];
int n,m;
int main()
{while(scanf("%d%d",&n,&m)!=EOF){memset(dp,0,sizeof(dp));memset(vis,0,sizeof(vis));memset(num,0,sizeof(num));for(int i=0;i<m;i++)scanf("%d",&num[i]);for(int i=0; i<=m-1; i++)//这里正序是错的!!倒序是对的!用上面给的那个样例试一下就知道了。sum输出的是对的但是路径是错的。 for(int j=n;j>=num[i];j--){if(dp[j-num[i]]+num[i]>dp[j]){dp[j]=dp[j-num[i]]+num[i];vis[i][j]=1;
//					printf("vis[%d][%d] = 1\n",i,j);}}
//		printf("vis[1]][n] = %d\n",vis[0][n]);int j=n;
//		for(int i = 1; i<=n; i++) {
//			printf("dp[%d] = %d\n",i,dp[i]);
//		}int cnt = 0;int ans[50] = {0};for(int i=m-1;i>=0;i--){if(vis[i][j]){ans[++cnt] = num[i];j-=num[i];}} for(int i = cnt; i>=1; i--) printf("%d ",ans[i]);printf("sum:%d\n",dp[n]);}return 0;
}

其实输出路径还有第三种写法,不针对这个题但是可以这样做:(还原成二维数组但是可以不开vis数组了)

这个题也可以这样做

45 3

38 38

39 39

2  2

这样输入就可以了。。。

其中x数组记录的是第i个物品在路径中是否用到了。

所以输出为:

0 1 1

 

#include<iostream>
#include<cstring>
using namespace std;
int main()
{int V,n;while(cin>>V>>n){int dp[100][100];int w[1000],v[1000];for(int i = 1;i <= n;i++)cin>>w[i]>>v[i];for(int i = 0;i <= V;i++)dp[0][i]=0;for(int i = 1;i <= n;i++){for(int j = V; j>=w[i];j--){dp[i][j]=max(dp[i-1][j],dp[i-1][j-w[i]]+v[i]);}}cout<<dp[n][V]<<endl;int j=V;int x[1000];memset(x,0,sizeof(x));for(int i = n;i >= 1;i--){if(dp[i][j]>dp[i-1][j]){x[i]=1;j-=w[i];}}for(int i = 1;i <= n;i++)cout<<x[i]<<" ";cout<<endl;}return 0;
}

总结:

   其实很多背包问题之所以需要排序(其实倒序做背包问题也算是排序的一种啦~),有的是为了路径的正确性(比如说这一题),但是也有可能就是会影响到结果的正确性(比如:POJ 2392 Space Elevator (多重背包),或者HDU - 3466 Proud Merchants(蜜汁排序) )。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/439634.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

详解两阶段3D目标检测网络 Voxel R-CNN:Towards High Performance Voxel-based 3D Object Detection

本文介绍一篇两阶段的3D目标检测网络&#xff1a;Voxel R-CNN&#xff0c;论文已收录于AAAI 2021。 这里重点是理解本文提出的 Voxel RoI pooling。 论文链接为&#xff1a;https://arxiv.org/pdf/2012.15712.pdf 项目链接为&#xff1a;https://github.com/djiajunustc/Voxe…

java容器类1:Collection,List,ArrayList,LinkedList深入解读

1、 Iterable 与 Iterator Iterable 是个接口&#xff0c;实现此接口使集合对象可以通过迭代器遍历自身元素. public interface Iterable<T> 修饰符和返回值方法名描述Iterator<T>iterator()返回一个内部元素为T类型的迭代器default voidforEach(Consumer<?…

【POJ - 1050】To the Max (dp)

题干&#xff1a; Given a two-dimensional array of positive and negative integers, a sub-rectangle is any contiguous sub-array of size 1*1 or greater located within the whole array. The sum of a rectangle is the sum of all the elements in that rectangle. I…

无限场景开放式仿真器 PGDrive:Improving the Generalization of End-to-End Driving through Procedural Generation

本文介绍一个拥有无限场景开放式驾驶仿真器&#xff1a;PGDrive&#xff0c;通过 Procedural Generation 技术可以生成无限多的驾驶场景&#xff0c;由香港中文大学周博磊团队开发。 论文地址&#xff1a;https://arxiv.org/pdf/2012.13681.pdf 项目地址&#xff1a;https://…

java容器类2:Map及HashMap深入解读

Java的编程过程中经常会和Map打交道&#xff0c;现在我们来一起了解一下Map的底层实现&#xff0c;其中的思想结构对我们平时接口设计和编程也有一定借鉴作用。(以下接口分析都是以jdk1.8源码为参考依据) 1. Map An object that maps keys to values. A map cannot contain du…

两阶段3D目标检测网络 SIENet: Spatial Information Enhancement Network for 3D Object Detection from Point Cloud

本文介绍一篇两阶段的3D目标检测网络&#xff1a;SIENet。 这里重点是理解本文提出的 Hybrid-Paradigm Region Proposal Network 和 Spatial Information Enhancement module。 论文链接为&#xff1a;https://arxiv.org/abs/2103.15396 项目链接为&#xff1a;https://githu…

java容器类3:set/HastSet/MapSet深入解读

介绍 Set&#xff1a;集合&#xff0c;是一个不包含重复数据的集合。&#xff08;A collection that contains no duplicate elements. &#xff09; set中最多包含一个null元素&#xff0c;否者包含了两个相同的元素&#xff0c;不符合定义。 上一篇学习了Java中的容器类的一…

Bandit算法原理及Python实战

目录 1&#xff09;什么是Bandit算法 为选择而生。 Bandit算法与推荐系统 怎么选择Bandit算法&#xff1f; 2)常用Bandit算法 Thompson sampling算法 UCB算法 Epsilon-Greedy算法 Greedy算法 3&#xff09;Bandit算法Python实战 参考资料&#xff1a; 推荐系统里面有…

*【 POJ - 1007 】DNA Sorting(枚举,类似三元组找第二元问题)

题干&#xff1a; One measure of unsortedness in a sequence is the number of pairs of entries that are out of order with respect to each other. For instance, in the letter sequence DAABEC, this measure is 5, since D is greater than four letters to its righ…

ava容器类4:Queue深入解读

Collection的其它两大分支&#xff1a;List和Set在前面已近分析过&#xff0c;这篇来分析一下Queue的底层实现。 前三篇关于Java容器类的文章&#xff1a; java容器类1&#xff1a;Collection,List,ArrayList,LinkedList深入解读 java容器类2&#xff1a;Map及HashMap深入解…

Waymo离线点云序列3D物体检测网络 (3D Auto Labeling): Offboard 3D Object Detection from Point Cloud Sequences

本文介绍一篇Waymo基于点云序列的3D物体检测网络&#xff1a;3D Auto Labeling&#xff0c;论文已收录于CVPR 2021。 这里重点是理解本文提出的 Object-centric Auto Labeling。 论文链接为&#xff1a;https://arxiv.org/abs/2103.05073 2021-09-02补充&#xff1a;本文作者…

【OpenJ_Bailian - 2711 】 合唱队形(dp,枚举中间顶点)

题干&#xff1a; N位同学站成一排&#xff0c;音乐老师要请其中的(N-K)位同学出列&#xff0c;使得剩下的K位同学不交换位置就能排成合唱队形。 合唱队形是指这样的一种队形&#xff1a;设K位同学从左到右依次编号为1, 2, …, K&#xff0c;他们的身高分别为T1, T2, …, TK&…

Waymo自动驾驶数据集介绍与使用教程

本文将对Waymo自动驾驶数据集&#xff08;Waymo Open Dataset&#xff09;进行介绍。 论文链接为&#xff1a;https://arxiv.org/abs/1912.04838v7 项目链接为&#xff1a;https://github.com/waymo-research/waymo-open-dataset 数据集链接为&#xff1a;https://waymo.com…

Java 并发基础——线程安全性

线程安全&#xff1a;多个线程访问某个类时&#xff0c;不管运行时环境采用何种调度方式或者这些线程将如何交替执行&#xff0c;并且在主调代码中不需要任何额外的同步或协调&#xff0c;这个类都能表现出正确的行为&#xff0c;那么久称这个类是线程安全的。 在线程安全类中封…

详解一阶段3D物体检测网络 SE-SSD: Self-Ensembling Single-Stage Object Detector From Point Cloud

本文介绍一篇一阶段的3D物体检测网络&#xff1a;SE-SSD&#xff0c;论文已收录于 CVPR 2021。 这里重点是理解本文提出的 Consistency Loss 、Orientation-Aware Distance-IoU Loss、Shape-Aware Data Augmentation。 论文链接为&#xff1a;https://arxiv.org/pdf/2104.0980…

【POJ - 3744】Scout YYF I(概率dp,矩阵快速幂优化dp)

题干&#xff1a; 题目大意&#xff1a; 在一条不满地雷的路上&#xff08;无限长&#xff09;&#xff0c;你现在的起点在1处。在N个点处布有地雷&#xff0c;1<N<10。地雷点的可能坐标范围&#xff1a;[1,100000000]. 每次前进p的概率前进一步&#xff0c;1-p的概率…

详解3D点云分割网络 Cylindrical and Asymmetrical 3D Convolution Networksfor LiDAR Segmentation

本文介绍一篇3D点云分割网络&#xff1a;Cylinder3D&#xff0c;论文已收录于 CVPR 2021。 这里重点是理解本文提出的 Cylindrical Partition 和 Asymmetrical 3D Convolution Network。 论文链接为&#xff1a;https://arxiv.org/pdf/2011.10033.pdf 项目链接为&#xff1a;…

Java中泛型Class《T》、T与Class《?》

一.区别 单独的T 代表一个类型 &#xff0c;而 Class<T>代表这个类型所对应的类&#xff0c; Class<&#xff1f;>表示类型不确定的类 E - Element (在集合中使用&#xff0c;因为集合中存放的是元素)T - Type&#xff08;Java 类&#xff09;K - Key&#xff08;…

【CodeForces - 701D】As Fast As Possible(二分,模拟,数学公式)

题干&#xff1a; On vacations n pupils decided to go on excursion and gather all together. They need to overcome the path with the length l meters. Each of the pupils will go with the speed equal to v1. To get to the excursion quickly, it was decided to r…

自动驾驶3D物体检测研究综述 3D Object Detection for Autonomous Driving: A Survey

本文介绍一篇最新的自动驾驶3D物体检测研究综述&#xff08;2021年6月份发布&#xff09;&#xff0c;来源于中国人民大学&#xff0c;论文共26页&#xff0c;99篇参考文献。 论文链接为&#xff1a;https://arxiv.org/pdf/2106.10823.pdf 0. Abstract 自动驾驶被看作是避免人…