算法设计与分析/数据结构与算法实验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;
}


目录
相关文章
|
数据采集 机器学习/深度学习 算法
别急着上算法,咱先把数据整明白:大数据分析的5个基本步骤,你都搞对了吗?
别急着上算法,咱先把数据整明白:大数据分析的5个基本步骤,你都搞对了吗?
929 4
|
12月前
|
机器学习/深度学习 边缘计算 算法
NOMA和OFDMA优化算法分析
NOMA和OFDMA优化算法分析
563 127
|
9月前
|
运维 监控 JavaScript
基于 Node.js 图结构的局域网设备拓扑分析算法在局域网内监控软件中的应用研究
本文探讨图结构在局域网监控系统中的应用,通过Node.js实现设备拓扑建模、路径分析与故障定位,提升网络可视化、可追溯性与运维效率,结合模拟实验验证其高效性与准确性。
479 3
|
9月前
|
存储 边缘计算 算法
【太阳能学报EI复现】基于粒子群优化算法的风-水电联合优化运行分析(Matlab代码实现)
【太阳能学报EI复现】基于粒子群优化算法的风-水电联合优化运行分析(Matlab代码实现)
178 0
|
11月前
|
编解码 算法 5G
MIMO雷达空间谱估计中Capon算法与MUSIC算法的对比分析及实现
MIMO雷达空间谱估计中Capon算法与MUSIC算法的对比分析及实现
946 2
|
11月前
|
人工智能 自然语言处理 算法
2025 年 7 月境内深度合成服务算法备案情况分析报告
2025年7月,中央网信办发布第十二批深度合成算法备案信息,全国389款产品通过备案,服务提供者占比超七成。截至7月14日,全国累计备案达3834款,覆盖文本、图像、音视频等多模态场景,广泛应用于生活服务、医疗、金融等领域。广东以135款居首,数字人、AI客服等C端应用主导,民营企业成主力,国企聚焦公共服务。随着AI政策推动,备案已成为AI产品合规上线关键环节。
|
10月前
|
机器学习/深度学习 算法 5G
【MUSIC、最大似然与克拉美-罗下界】MUSIC与ESPRIT 算法来估计到达角(AoA),并尝试推导克拉美-罗下界(CRLB)以分析其性能研究(Matlab代码实现)
【MUSIC、最大似然与克拉美-罗下界】MUSIC与ESPRIT 算法来估计到达角(AoA),并尝试推导克拉美-罗下界(CRLB)以分析其性能研究(Matlab代码实现)
687 0
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
416 59
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
1076 77
|
编译器 C语言 C++
栈区的非法访问导致的死循环(x64)
这段内容主要分析了一段C语言代码在VS2022中形成死循环的原因,涉及栈区内存布局和数组越界问题。代码中`arr[15]`越界访问,修改了变量`i`的值,导致`for`循环条件始终为真,形成死循环。原因是VS2022栈区从低地址到高地址分配内存,`arr`数组与`i`相邻,`arr[15]`恰好覆盖`i`的地址。而在VS2019中,栈区先分配高地址再分配低地址,因此相同代码表现不同。这说明编译器对栈区内存分配顺序的实现差异会导致程序行为不一致,需避免数组越界以确保代码健壮性。
254 0
栈区的非法访问导致的死循环(x64)