UVa CD 0-1背包且打印路径

2024-06-19 15:32
文章标签 uva cd 背包 打印 路径

本文主要是介绍UVa CD 0-1背包且打印路径,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

       就是简单的0-1背包问题,不过没有具体的效益值,隐含的效益值就是剩余背包的容量。因为要输出具体选择了那些track(也就是物品),所以采用序偶的方法。其实0-1背包的解画在坐标轴上就是一个分段函数,所谓序偶就是那些跃迁的节点。但是这道题略有不同,第0阶段的初始序偶不是(0,0),而是(0,N)。序偶的第一个参数表示容量,第二个参数表示背包的剩余容量。当由前一阶段的序偶得到新序偶,并且将两者合并的时候。应当按照序偶的第一个参数从小到大进行合并,如果下一个要合并的序偶的剩余容量大于当前最后一个以合并序偶的剩余容量要大,则将该序偶舍去。因为更大的容量,但是却有更多的剩余容量是不合理的。(容量比前者大,至少把前者装的东西都能装下)。

       最后是打印输出。做法是从最后一阶段开始,若当前序偶在前一阶段存在,则不打印,否则找到前一阶段中,减去该物品的序偶,重复前面的过程。说的可能有点抽象....还是结合代码看吧~

#include <iostream>
#include <vector>
#include <utility>
#include <cstdio>
#define MAX 20+5
using namespace std;int N,CD;
int track[MAX];
vector <pair<int,int> > ans[MAX];
void print(int,int);int main()
{while(cin>>N>>CD){for(int i=1;i<=CD;i++) cin>>track[i];for(int i=0;i<=CD;i++) ans[i].clear();ans[0].push_back(make_pair(0,N));for(int i=1;i<=CD;i++){vector <pair<int,int> > temp;for(auto iter=ans[i-1].begin();iter!=ans[i-1].end();iter++)//坐标平移得到新序偶temp.push_back(make_pair(iter->first+track[i],iter->second-track[i]));int x=0,y=0;while(x<ans[i-1].size()&&y<temp.size()){//按照容量从小到大合并序偶//剩余容量比前一个已添加序偶的剩余容量还大,则删除此序偶if(ans[i-1][x].first<temp[y].first){if(ans[i].empty()||ans[i-1][x].second<ans[i][ans[i].size()-1].second)ans[i].push_back(ans[i-1][x]);x++;}else if(ans[i-1][x].first>temp[y].first){if(ans[i].empty()||temp[y].second<ans[i][ans[i].size()-1].second)ans[i].push_back(temp[y]);y++;}else{int Min=min(ans[i-1][x].second,temp[y].second);if(ans[i].empty()||Min<ans[i][ans[i].size()-1].second)ans[i].push_back(make_pair(temp[y].first,Min));x++,y++;}}while(x<ans[i-1].size()){//合并剩余序偶if(ans[i-1][x].second<ans[i][ans[i].size()-1].second)ans[i].push_back(ans[i-1][x]);x++;}while(y<temp.size()){if(temp[y].second<ans[i][ans[i].size()-1].second)ans[i].push_back(temp[y]);y++;}}int pos;for(pos=0;pos<ans[CD].size();pos++)//找到第一个容量比给定N大的序偶if(ans[CD][pos].first>N) break;pos--;int sum=N-ans[CD][pos].second;print(CD,pos);//递归输出路径cout<<"sum:"<<sum<<endl;}return 0;
}void print(int depth,int pos)
{if(depth==0) return;for(int i=0;i<ans[depth-1].size();i++)if(ans[depth-1][i].first==ans[depth][pos].first-track[depth]&&ans[depth-1][i].second==ans[depth][pos].second+track[depth]){print(depth-1,i);cout<<track[depth]<<" ";break;}else if(ans[depth-1][i].first==ans[depth][pos].first&&ans[depth-1][i].second==ans[depth][pos].second){print(depth-1,i);break;}return ;
}


这篇关于UVa CD 0-1背包且打印路径的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1075421

相关文章

随想录 Day 69 并查集 107. 寻找存在的路径

随想录 Day 69 并查集 107. 寻找存在的路径 理论基础 int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好vector<int> father = vector<int> (n, 0); // C++里的一种数组结构// 并查集初始化void init() {for (int i = 0; i < n; ++i) {father[i] = i;}

SQL Server中,用Restore DataBase把数据库还原到指定的路径

restore database 数据库名 from disk='备份文件路径' with move '数据库文件名' to '数据库文件放置路径', move '日志文件名' to '日志文件存放置路径' Go 如: restore database EaseWe from disk='H:\EaseWe.bak' with move 'Ease

C# 命名管道中客户端访问服务器时,出现“对路径的访问被拒绝”

先还原一下我出现错误的情景:我用C#控制台写了一个命名管道服务器,然后用ASP.NET写了一个客户端访问服务器,运行之后出现了下面的错误: 原因:服务器端的访问权限不够,所以是服务器端的问题,需要增加访问权限。(网上很多都说是文件夹的权限不够,情况不同,不适用于我这种情况) 解决办法: (1)在服务器端相应地方添加以下代码。 PipeSecurity pse = new PipeSec

代码随想录算法训练营第三十九天|62.不同路径 63. 不同路径 II 343.整数拆分 96.不同的二叉搜索树

LeetCode 62.不同路径 题目链接:62.不同路径 踩坑:二维的vector数组需要初始化,否则会报错访问空指针 思路: 确定动态数组的含义:dp[i][j]:到达(i,j)有多少条路经递推公式:dp[i][j] = dp[i-1][j] + dp[i][j-1]初始化动态数组:dp[0][0] = 1遍历顺序:从左到右,从上到下 代码: class Solution {pu

Android Log日志 - 打印不全问题

AndroidStudio在打印Log的时候目前支持4*1024长度,超出部分不能打印。当你在各种百度之后有对应的解决办法,但是每次都是部分代码,看着都忧伤。索性此次项目调试的数据也是比较多滴,目前就准备对Log开刀来写一个Log类,还是如以往的性格直接写完整的类,方便需要的人用。反正又不是什么高深的东西,为了给被方便同时也是给自己方便。 /*** Relin* 2019-07-10 10:40

UVA - 12206 Stammering Aliens (hash)

这题其实很容易想到2分长度,关键是2分后,怎么判断出现最多次的串是否》=m次了。这里需要用到hash来处理。 对于一个字符串   H[i] =H[i+1]*131+s[i]  ;其中 H[n]=0;那么对于一个长度为L的串 ,hanh[i,l]=H[i]-H[i+L]*x^L ; 这样记录每个字符串的hash值,然后再处理起来就比较简单了。 VIEW CODE #include<

秋招突击——6/22——复习{区间DP——加分二叉树,背包问题——买书}——新作{移除元素、实现strStr()}

文章目录 引言复习区间DP——加分二叉树个人实现 背包问题——买书个人实现参考实现 新作移除元素个人实现参考思路 找出字符串中第一个匹配项的下标个人实现参考实现 总结 引言 今天做了一个噩梦,然后流了一身汗,然后没起来,九点多才起床背书。十点钟才开始把昨天那道题题目过一遍,然后十一点才开始复习题目,为了不耽误下午的时间,所以这里的就单纯做已经做过的题目,主打一个有量,不在学

Android gradle打印依赖的各种姿势

查看Android Gradle 依赖树 1.查看单独模块的依赖 命令行 ./gradlew :模块名:dependencies 例子: ./gradlew :app:dependencies 这个命令会将 gradle 执行的各个步骤全打印出来,包括引用的库,和库中引用的库文件 ./gradlew :app:dependencies --configuration im

WinCE的C#程序中获取当前应用程序的路径

WinCE中获取当前路径的两种方法: string appPath = System.IO.Path.GetDirectoryName(System.Reflection.Assembly.GetExecutingAssembly().GetName().CodeBase); string appPath = System.IO.Path.GetDirectoryName(System.R