Fork me on GitHub
千里山水藏于心,你藏山水里 ~~~

回溯之0-1背包

0-1背包问题:物品总数n,每个物品的体积w[i],价值v[i],给定背包的总容量W,求放入背包中物品的最大价值。0-1背包回溯背景和思路是这样的:

 

用回溯法对0-1背包问题进行求解,具体思路是:

1.使用解空间进行标记每个物品的放入情况,即要建立一个数组进行保存其是否放入,可使用 bool  x[i]进行标识;

2.回溯法第一感觉上是穷举所有情况,但事实上,有好多种情况可以进行避免,即若第t个物品放入后(即x[t]=1)已经超出背包重量,那么,在x[t]=1情况下的t+1—n个物品就不用再考虑,这样可以节省好多时间,回溯法有区别与穷举法;

3.对于解空间,用解空间数进行组织数据,解空间树的深度就是问题的规模n;

4.在解空间树中,我们用左子树、右子树分别标记1/0情况,即左子树的边代表放入,右子树的边代表不放入;

5.建立回溯函数是重中之重,回溯函数建立有三步(判断是否到达叶节点,约束条件(这里是基于重量),限界剪枝(剩余价值是否大于最优,否则没有算下去的必要了))

具体的实现如下:

 1 #include <iostream>
 2 #define N 100
 3 using namespace std;
 4 int n;
 5 double W;
 6 double w[N];
 7 double v[N];
 8 bool x[N];  //用于记录某次回溯情况
 9 bool best_x[N]; //存储最优回溯情况
10 double now_v;   //当前价值
11 double remain_v;    //剩余价值
12 double now_w;   //当前容量
13 double best_v;  //最优价值
14 
15 //计算剩余价值
16 double Bound(int k){
17     remain_v = 0;
18     while (k <= n) {
19         remain_v += v[k];
20         k++;
21     }
22     return remain_v + now_v;
23 }
24 //剪枝的三个判定
25 void Backtrack(int t){
26     if (t > n) {  //是否到达叶节点
27         for (int i = 1; i <= n; i++) {
28             best_x[i] = x[i];   //记录回溯的最优情况
29         }
30         best_v = now_v; //记录回溯中的最优价值
31         return;
32     }
33     if (now_w + w[t] <= W) {  //约束条件,是否放入。放入考虑左子树,否则考虑右子树
34         x[t] = 1;
35         now_w += w[t];
36         now_v += v[t];
37         Backtrack(t + 1); //进行下一个节点的分析
38         now_w -= w[t];  //在到达叶节点后进行回溯
39         now_v -= v[t];
40     }
41     if (Bound(t + 1) > best_v) {    //限界条件,是否剪枝。若放入t后不满足约束条件则进行到此处,然后判断若当前价值加剩余价值都达不到最优,则没必要进行下去
42         x[t] = 0;
43         Backtrack(t + 1);
44     }
45 }
46 //实现价值最优
47 void Knapsack(double W, int n){
48     double sum_w = 0;
49     double sum_v = 0;
50     best_v = 0;
51     for (int i = 0; i < n; i++) {
52         sum_w += w[i];
53         sum_v += v[i];
54     }
55     if (sum_w <= W) {
56         best_v = sum_v;
57         cout << "These goods could be put in the shopping car";
58         cout << "The best value is:" << best_v << endl;
59         return;
60     }
61     Backtrack(1);
62     cout << "The best value is:" << best_v << endl;
63     cout << "The condiction of these goods are:" << endl;
64     for (int i = 1; i <= n; i++) {
65         cout << i << " ";
66         cout << best_x[i] << endl;   //打印所有物品的放入情况,若为1表示放入,若为0则表示不放入
67     }
68 }
69 int main(){
70     cout << "Please input the num of goods:";
71     cin >> n;
72     cout << "Please input the capacity of the shopping car:";
73     cin >> W;
74     cout << "Please input thier weights and values in order:" << endl;
75     for (int i = 1; i <= n; i++) {
76         cin >> w[i] >> v[i];
77     }
78     Knapsack(W, n);
79     return 0;
80 }
posted @ 2019-06-20 10:03  3if  阅读(162)  评论(0编辑  收藏  举报