算法设计与分析/数据结构与算法实验7:0-1背包问题(分支限界法)

简介: 算法设计与分析/数据结构与算法实验7:0-1背包问题(分支限界法)

1.实验目的

 (1)掌握分支限界法的处理思路与算法框架。

 (2)掌握应用分支限界法解决具体问题的方法。

 (3)掌握分支限界法的广泛应用。

2.实验内容

(1)问题描述

image.png


 要求使用分支限界法解决该问题。

(2)输入

 输入包含3行。

image.png


(3)输出

image.png


3.问题实例分析

image.png


 堆(优先队列)是大根堆,“优先级”即判断函数为价值上界。

 此时大根堆堆顶为(“拿1,上界14.33”)。对其进行扩展,以是否拿2作为区分的依据。拿2时,价值上界为14.33。不拿2时,价值上界为14。将这两个结点都加入堆(优先队列)。

 此时大根堆堆顶为“拿2,上界14.33”。对其进行扩展,以是否拿3作为区分依据。拿3时,价值上界为14.33。不拿3时,价值上界为14。将这两个结点都加入堆(优先队列)。

 此时大根堆堆顶为“拿3,上界14.33”。对其进行扩展,以是否拿4作为区分依据。拿4时,背包此时体积超过上限,不可行。不拿4时,价值上界为14.25。将可行的右结点加入堆(优先队列)。

 此时大根堆堆顶为“不拿4,上界14.25”。对其进行扩展,以是否拿5作为区分依据。拿5时,背包此时体积超过上限,不可行。不拿5时,价值上界为13。将可行的右结点加入堆(优先队列)。

 此时大根堆堆顶为“不拿3,上界14”。对其进行扩展,以是否拿4作为区分依据。拿4时,价值上界为14。不拿4时,价值上界为10。这两个结点都可行,都加入堆(优先队列)。

 此时大根堆堆顶为“拿4,上界14”。对其进行扩展,以是否拿5作为区分依据。拿5时,价值上界为14。不拿5时,价值上界为9。这两个结点都可行,都加入堆(优先队列)。

image.png

4.算法描述及说明

 正如第3节问题实例分析所述,算法的整体流程如下:

 1.输入数据,并对每个物品进行编号。

 2.计算每个物品的单位价值,并将物品按单位价值排序。

 3.计算根结点上界,并将根结点加入大根堆。

 4.取出堆顶元素。若堆顶结点为解空间树的叶结点,则算法直接结束,叶结点的物品选取信息和价值为所求的最大价值。

 5.若堆顶元素不是叶结点,则尝试扩展左结点。若左结点可行(选取该物品背包体积没有被超出),则对左结点计算上界后入堆。扩展右结点,不选取物品,对右结点计算上界后入堆。

5.算法正确性分析

 算法会正确地结束:在利用优先队列遍历解空间后,找到了使总价值最大的解,算法会停止。

 分支限界的正确性分析:开始时,根结点是解空间树唯一的活结点,也是当前的扩展结点。算法会不断扩展结点,直到子集树的一个叶结点成为扩展结点时为止。此时优先队列中所有活结点的价值上界不超过该叶结点的价值。因此,该叶结点对应的解为问题最优解。

 因此,利用分支限界法会系统地查找背包问题的所有可行解,利用限界函数剪去了不可行的分支,保留了可行并能产生最大解的分支。

 从而,该算法是正确的。

6.算法时间复杂性分析

image.png


7.运行结果展示及其说明

image.png

8.心得体会

9.程序源代码

#include<iostream>
#include<algorithm>
#include<queue>
using namespace std;
double cw;//当前重量
double cp;//当前价值
double bestp;//当前最优价值
int n;//物品数量
double c;//背包容量
const int N = 105;
struct Bag {
  double w, v;
  int id, x;
};
int bestx[N];
Bag bag[N];
struct BBnode {
  BBnode *parent;
  bool leftChild;
  BBnode(BBnode* par, bool ch) {
    parent = par;
    leftChild = ch;
  }
};
struct heapnode {
  BBnode* livenode;
  double upperprofit;//价值上界
  double profit;//结点价值
  double weight;//结点重量
  int level;//层序号
  bool operator<(const heapnode& b) const {
    return this->upperprofit < b.upperprofit;
  }
  heapnode(BBnode *node,double up, double pp, double ww, int lev) {
    livenode = node;
    upperprofit = up;
    profit = pp;
    weight = ww;
    level = lev;
  }
};
bool cmp(Bag a, Bag b) {
  return (a.v / a.w) > (b.v / b.w);
}
bool cmpbound(heapnode a, heapnode b) {
  return a.upperprofit < b.upperprofit;
}
priority_queue<heapnode, vector<heapnode>,less<heapnode> > q;
void addlivenode(double up, double pp, double ww, int lev, BBnode* par, bool ch) {
  BBnode* b = new BBnode(par, ch);
  heapnode node(b, up, pp, ww, lev);
  q.push(node);
}
double bound(int i) {
  double cleft = c - cw;
  double bd = cp;
  while (i <= n && bag[i].w <= cleft) {
    cleft -= bag[i].w;
    bd += bag[i].v;
    i++;
  }
  if (i <= n)
    bd += bag[i].v * cleft / bag[i].w;
  return bd;
}
void bfs() {
  BBnode *enode = NULL;
  int i = 1;
  bestp = 0;
  double up = bound(1);
  while (i != n + 1) {
    double wt = cw + bag[i].w;
    if (wt <= c) {
      if (cp + bag[i].v > bestp)
        bestp = cp + bag[i].v;
      addlivenode(up, cp + bag[i].v, cw + bag[i].w, i + 1, enode, true);
    }
    up = bound(i + 1);
    if (up > bestp)
      addlivenode(up, cp, cw, i + 1, enode, false);
    heapnode node = q.top();
    q.pop();
    enode = node.livenode;
    cw = node.weight;
    cp = node.profit;
    up = node.upperprofit;
    i = node.level;
  }
  for (int j = n; j > 0; j--) {
    bag[j].x = (enode->leftChild!=NULL) ? 1 : 0;
    enode = enode->parent;
  }
  for (int i = 1; i <= n; i++)
    bestx[bag[i].id] = bag[i].x;
  return;
}
int main() {
  cin >> n >> c;
  for (int i = 1; i <= n; i++)
    cin >> bag[i].w;
  for (int i = 1; i <= n; i++)
    cin >> bag[i].v;
  for (int i = 1; i <= n; i++)
    bag[i].id = i;
  sort(bag + 1, bag + 1 + n, cmp);
  bfs();
  cout << bestp << endl;
  for (int i = 1; i <= n; i++)
    cout << bestx[i] << " ";
  return 0;
}


目录
打赏
0
0
0
0
54
分享
相关文章
|
28天前
|
算法系列之数据结构-二叉树
树是一种重要的非线性数据结构,广泛应用于各种算法和应用中。本文介绍了树的基本概念、常见类型(如二叉树、满二叉树、完全二叉树、平衡二叉树、B树等)及其在Java中的实现。通过递归方法实现了二叉树的前序、中序、后序和层次遍历,并展示了具体的代码示例和运行结果。掌握树结构有助于提高编程能力,优化算法设计。
47 9
 算法系列之数据结构-二叉树
|
25天前
|
算法系列之数据结构-Huffman树
Huffman树(哈夫曼树)又称最优二叉树,是一种带权路径长度最短的二叉树,常用于信息传输、数据压缩等方面。它的构造基于字符出现的频率,通过将频率较低的字符组合在一起,最终形成一棵树。在Huffman树中,每个叶节点代表一个字符,而每个字符的编码则是从根节点到叶节点的路径所对应的二进制序列。
40 3
 算法系列之数据结构-Huffman树
|
27天前
|
算法系列之数据结构-二叉搜索树
二叉查找树(Binary Search Tree,简称BST)是一种常用的数据结构,它能够高效地进行查找、插入和删除操作。二叉查找树的特点是,对于树中的每个节点,其左子树中的所有节点都小于该节点,而右子树中的所有节点都大于该节点。
65 22
C 408—《数据结构》算法题基础篇—链表(下)
408考研——《数据结构》算法题基础篇之链表(下)。
97 29
C 408—《数据结构》算法题基础篇—链表(上)
408考研——《数据结构》算法题基础篇之链表(上)。
127 25
C 408—《数据结构》算法题基础篇—数组(通俗易懂)
408考研——《数据结构》算法题基础篇之数组。(408算法题的入门)
84 23
探索局域网电脑监控软件:Python算法与数据结构的巧妙结合
在数字化时代,局域网电脑监控软件成为企业管理和IT运维的重要工具,确保数据安全和网络稳定。本文探讨其背后的关键技术——Python中的算法与数据结构,如字典用于高效存储设备信息,以及数据收集、异常检测和聚合算法提升监控效率。通过Python代码示例,展示了如何实现基本监控功能,帮助读者理解其工作原理并激发技术兴趣。
95 20
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
79 2
Python 中的数据结构和算法优化策略
Python中的数据结构和算法如何进行优化?
102 19
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
115 1

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等