算法设计与分析——分支限界法

1. 分支界限法

1.1 基本思想

对有约束条件的最优化问题的所有可行解(数目有限)空间进行搜索。该算法在具体执行时,把全部可行的解空间不断分割为越来越小的子集(称为分支),并为每个子集内的解的值计算一个下界或上界(称为界限)。在每次分支后,对凡是界限超出已知可行解值的那些子集不在做进一步分支。这样就缩小了搜索范围。这一过程一直进行到找出可行解为止,该可行解的值不大于任何子集的界限。

1.2 搜索策略

在扩展结点处,先生成其所有的儿子结点(分支),然后再从当前的活结点表中选择下一个扩展结点。为了有效地选择下一扩展结点,加速搜索的进程,在每一个活结点处,计算一个函数值(限界),并根据函数值,从当前活结点表中选择一个最有利的结点作为扩展结点,使搜索朝着解空间上有最优解的分支推进,以便尽快地找出一个最优解。

1.3 队列式

按照队列先进先出(FIFO)原则选取下一个节点为扩展节点。

1.4* 优先队列式

按照优先队列中规定的优先级选取优先级最高的节点成为当前扩展节点。

2. 典型案例

2.1 装载问题

2.1.1 问题描述

集装箱装载问题要求确定在不超过轮船载重量的前提下,将尽可能多的集装箱装上轮船。

2.1.2 约束函数和限界条件

约束函数:当 Ew+wi > c 对扩展结点的左子树剪枝

限界函数:当 Ew+r <= bestw 对扩展结点的右子树剪枝

2.1.3 队列式分支界限法求解(案例解释)

轮船的载重量为c=80、集装箱个数n=4、重量分别为18 7 25 36。

定义一个先进先出(FIFO)队列Q,初始化队列时,在尾部增加一个 -1 标记。这是一个分层的标志,当一层结束时,在队列尾部增加一个 -1 标志。

定义扩展结点相应的载重量为Ew,剩余集装箱的重量为r,当前最优载重量为bestw

没到一个结点就要计算bestw、r、Ew,然后进行约束函数和限界函数的判断,如果不满足约束函数则对左子树剪枝,不满足限界函数对右子树进行剪枝。例如下图中的结点M,此时bestw=50,r=7+36=43,r<bestw,则结点M不入队。结点O也是同样。

image-20210609134031901

2.1.4 案列二

image-20210609155430662

  • 基于队列式的分支限界法解决

image-20210609155804268

2.1. 代码实现

#include "iostream"
#include "queue"
using namespace std;

#define NUM 100
int n;			//集装箱的数量
int c;			//轮船的载重量
int w[NUM];		//集装箱的重量数组

int MaxLoading()
{
    queue<int> Q;
    Q.push(-1);
    int i = 0;
    int Ew = 0;
    int bestw = 0;
    int r = 0;
    for(int j=1; j<n; j++)
        r += w[j];
    //搜索子空间树
    while (true) {
        //检查左子树
        int wt = Ew + w[i];
        if (wt <= c) {      //检查约束条件
            if (wt > bestw) bestw = wt;
            //加入活结点队列
            if (i < n - 1) Q.push(wt);
        }
        //检查右子树
        //检查上界条件
        if (Ew + r > bestw && i < n - 1)
            Q.push(Ew);
        //从队列中取出活结点
        Ew = Q.front();
        Q.pop();
        if (Ew == -1) {    //判断同层的尾部
            if (Q.empty()) return bestw;
            //同层结点尾部标志
            Q.push(-1);
            //从队列中取出活结点
            Ew = Q.front();
            Q.pop();
            i++;
            r -= w[i];
        }
    }
    return bestw;
}
int main(){
    cin >> c;
    cin >> n;
    for (int i = 0; i < n; ++i) {
        cin >> w[i];
    }
    cout << MaxLoading();
    return 0;
}

2.2 0-1背包问题(基于优先队列)

2.2.1 队列的进出过程

每到一个结点就计算其上界,判断该上界值是否小于当前最优解,如果小于就减去,如果不小于就按照优先级(上界大小),按次序插入队列。

2.2.2 上界值的计算

image-20210613165751038

采用 贪心算法 计算结点的上界值:将剩余物品依其单位重量价值排序,然后依次装入物品,直至装不下时,再装入该物品的一部分而装满背包时的价值。

3. 回溯法与分支限界法异同

  • 相同点:

都是一种在问题的解空间树T中搜索问题解的算法。

  • 不同点:

(1)求解目标不同

(2)搜索方式不同

(3)对扩展结点的扩展方式不同

(4)存储空间的要求不同

搜索方式求解目标扩展方式存储空间
回溯法深度优先满足条件的所有解当探索到某一结点时,要先判断该结点是否包含问题的解,如果包含,就从该结点出发继续探索下去,如果该结点不包含问题的解,则逐层向其祖先结点回溯。
分支限界法广度优先满足条件一个解或者特定意义的最优解每个结点只有一次成为活结点的机会一旦成为扩展结点,就一次性产生所有儿子结点,并舍弃不可行解或非最优解的子结点。队列、优先队列

c++中 friend的用法

  1. 普通的非成员函数友元

这类友元函数最常见,通常是操作符。

  1. 类作为友元

类作为友元需要注意的是友元类和原始类之间的相互依赖关系,如果在友元类中定义的函数使用到了原始类的私有变量,那么就需要在友元类定义的文件中包含原始类定义的头文件。但是在原始类的定义中(包含友元类声明的那个类),就不需要包含友元类的头文件.
另外,不需要在类定义前去声明友元类,因为友元类的声明自身就是一种声明。

  1. 类成员函数作为友元函数

这个稍微有点复杂,因为你要类成员函数作为友元,你在声明友元的时候要用类限定符,所以必须先定义包含友元函数的类,但是在定义友元的函数时候,又必须事先定义原始类。通常的做法先定义包含友元函数的类,再定义原始类,这个顺序不能乱。

friend用法参考链接

员函数作为友元,你在声明友元的时候要用类限定符,所以必须先定义包含友元函数的类,但是在定义友元的函数时候,又必须事先定义原始类。通常的做法先定义包含友元函数的类,再定义原始类,这个顺序不能乱。

friend用法参考链接

  • 3
    点赞
  • 60
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,我可以为你介绍一下分支限界法和装载问题的相关内容。 分支限界法是一种求解最优化问题的算法,通过不断扩展解空间,逐步缩小搜索范围,最终找到最优解。它的核心思想是将问题划分成许多子问题,并采用优先队列(或优先级队列)来维护待扩展的子问题集合,每次取出优先级最高的子问题进行扩展,直到找到最优解或者队列为空。 而装载问题是一种典型的分支限界法应用场景,它的主要思想是在给定的一些物品中选出尽可能多的物品放入容量为C的背包中,使得背包中物品的总重量不超过C,并且背包中物品的总价值最大。这个问题可以通过分支限界法来求解。 下面是一个简单的 Java 代码实现,用于解决装载问题: ```java import java.util.*; public class BranchAndBound { public static void main(String[] args) { int[] w = {5, 10, 20, 30}; // 物品的重量 int[] v = {50, 60, 140, 120}; // 物品的价值 int C = 50; // 背包的容量 int n = w.length; // 物品的数量 int[] x = new int[n]; // 记录每个物品是否被选中 PriorityQueue<Node> queue = new PriorityQueue<>(); queue.offer(new Node(-1, 0, 0)); // 将根节点加入队列中 while (!queue.isEmpty()) { Node node = queue.poll(); // 取出优先级最高的子问题 if (node.level == n - 1) { // 如果是叶子节点,更新最优解 for (int i = 0; i < n; i++) { x[i] = node.x[i]; } } else { int level = node.level + 1; int weight = node.weight; int value = node.value; if (weight + w[level] <= C) { // 左子节点表示选中当前物品 int[] left = Arrays.copyOf(node.x, n); left[level] = 1; queue.offer(new Node(level, weight + w[level], value + v[level], left)); } // 右子节点表示不选当前物品 queue.offer(new Node(level, weight, value, node.x)); } } int max = 0; for (int i = 0; i < n; i++) { if (x[i] == 1) { System.out.println("第" + (i + 1) + "个物品被选中"); max += v[i]; } } System.out.println("最大价值为:" + max); } // 子问题节点 static class Node implements Comparable<Node> { int level; // 当前节点所在的层级 int weight; // 当前节点的背包重量 int value; // 当前节点的背包价值 int[] x; // 记录每个物品是否被选中 public Node(int level, int weight, int value) { this.level = level; this.weight = weight; this.value = value; this.x = new int[0]; } public Node(int level, int weight, int value, int[] x) { this.level = level; this.weight = weight; this.value = value; this.x = x; } @Override public int compareTo(Node o) { return o.value - this.value; // 根据价值进行优先级比较 } } } ``` 希望这个简单的例子能帮助你更好地理解分支限界法和装载问题。如果你还有其他问题或者疑惑,欢迎随时向我提出。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

krain.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值