分支限界法—單源最短路徑問題

轉自:http://www.cnblogs.com/chinazhangjie/archive/2010/11/01/1866136.htmlhtml

分支限界法與回溯法

  (1)求解目標:回溯法的求解目標是找出解空間樹中知足約束條件的全部解,而分支限界法的求解目標則是找出知足約束條件的一個解,或是在知足約束條件的解中找出在某種意義下的最優解。 
  (2)搜索方式的不一樣:回溯法以深度優先的方式搜索解空間樹,而分支限界法則以廣度優先或以最小耗費優先的方式搜索解空間樹。 

分支限界法的基本思想 node

  分支限界法常以廣度優先或以最小耗費(最大效益)優先的方式搜索問題的解空間樹。
  在分支限界法中,每個活結點只有一次機會成爲擴展結點。活結點一旦成爲擴展結點,就一次性產生其全部兒子結點。在這些兒子結點中,致使不可行解或致使非最優解的兒子結點被捨棄,其他兒子結點被加入活結點表中。
   此後,從活結點表中取下一結點成爲當前擴展結點,並重覆上述結點擴展過程。這個過程一直持續到找到所需的解或活結點表爲空時爲止。

常見的兩種分支限界法ios

  (1)隊列式(FIFO)分支限界法
      按照隊列先進先出(FIFO)原則選取下一個結點爲擴展結點。 
  (2)優先隊列式分支限界法
      按照優先隊列中規定的優先級選取優先級最高的結點成爲當前擴展結點。

1、單源最短路徑問題

1、問題描述 算法

  在下圖所給的有向圖G中,每一邊都有一個非負邊權。要求圖G的從源頂點s到目標頂點t之間的最短路徑。數組

  下圖是用優先隊列式分支限界法解有向圖G的單源最短路徑問題產生的解空間樹。其中,每個結點旁邊的數字表示該結點所對應的當前路長。ide

 

找到一條路徑:測試

目前的最短路徑是8,一旦發現某個結點的下界不小於這個最短路進,則剪枝:spa

同一個結點選擇最短的到達路徑:3d

2.剪枝策略code

  在算法擴展結點的過程當中,一旦發現一個結點的下界不小於當前找到的最短路長,則算法剪去以該結點爲根的子樹。

  在算法中,利用結點間的控制關係進行剪枝。從源頂點s出發,2條不一樣路徑到達圖G的同一頂點。因爲兩條路徑的路長不一樣,所以能夠將路長長的路徑所對應的樹中的結點爲根的子樹剪去。

3.算法思想

  解單源最短路徑問題的優先隊列式分支限界法用一極小堆來存儲活結點表。其優先級是結點所對應的當前路長。

算法從圖G的源頂點s和空優先隊列開始。結點s被擴展後,它的兒子結點被依次插入堆中。此後,算法從堆中取出具備最小當前路長的結點做爲當前擴展結點,並依次檢查與當前擴展結點相鄰的全部頂點。若是從當前擴展結點i到頂點j有邊可達,且從源出發,途經頂點i再到頂點j的所相應的路徑的長度小於當前最優路徑長度,則將該頂點做爲活結點插入到活結點優先隊列中。這個結點的擴展過程一直繼續到活結點優先隊列爲空時爲止。

實現

  1 2.剪枝策略在算法擴展結點的過程當中,一旦發現一個結點的下界不小於當前找到的最短路長,則算法剪去以該結點爲根的子樹。   在算法中,利用結點間的控制關係進行剪枝。從源頂點s出發,2條不一樣路徑到達圖G的同一頂點。因爲兩條路徑的路長不一樣,所以能夠將路長長的路徑所對應的樹中的結點爲根的子樹剪去。 3.算法思想解單源最短路徑問題的優先隊列式分支限界法用一極小堆來存儲活結點表。其優先級是結點所對應的當前路長。算法從圖G的源頂點s和空優先隊列開始。結點s被擴展後,它的兒子結點被依次插入堆中。此後,算法從堆中取出具備最小當前路長的結點做爲當前擴展結點,並依次檢查與當前擴展結點相鄰的全部頂點。若是從當前擴展結點i到頂點j有邊可達,且從源出發,途經頂點i再到頂點j的所相應的路徑的長度小於當前最優路徑長度,則將該頂點做爲活結點插入到活結點優先隊列中。這個結點的擴展過程一直繼續到活結點優先隊列爲空時爲止。實現
  2 /* 主題:單源最短路徑問題
  3 * 做者:chinazhangjie
  4 * 郵箱:chinajiezhang@gmail.com
  5 * 開發語言:C++
  6 * 開發環境:Mircosoft Virsual Studio 2008
  7 * 時間: 2010.11.01
  8 */
  9 
 10 #include <iostream>
 11 #include <vector>
 12 #include <queue>
 13 #include <limits>
 14 using namespace std;
 15 
 16 struct node_info
 17 {
 18 public:
 19     node_info (int i,int w) 
 20         : index (i), weight (w) {}
 21     node_info () 
 22         : index(0),weight(0) {}
 23     node_info (const node_info & ni) 
 24         : index (ni.index), weight (ni.weight) {}
 25 
 26     friend 
 27     bool operator < (const node_info& lth,const node_info& rth) {
 28         return lth.weight > rth.weight ; // 爲了實現從小到大的順序
 29     }
 30 
 31 public:
 32     int index; // 結點位置
 33     int weight; // 權值
 34 };
 35 
 36 struct path_info 
 37 {
 38 public:
 39     path_info ()
 40         : front_index(0), weight (numeric_limits<int>::max()) {}
 41 
 42 public:
 43     int front_index;
 44     int weight;
 45 };
 46 
 47 // single source shortest paths
 48 class ss_shortest_paths
 49 {
 50      
 51 public:
 52     ss_shortest_paths (const vector<vector<int> >& g,int end_location) 
 53         :no_edge (-1), end_node (end_location), node_count (g.size()) , graph (g) 
 54     {}
 55 
 56     // 打印最短路徑
 57     void print_spaths () const {
 58         cout << "min weight : " << shortest_path << endl;
 59         cout << "path: " ;
 60         copy (s_path_index.rbegin(),s_path_index.rend(),
 61             ostream_iterator<int> (cout, " "));
 62         cout << endl;
 63     }
 64 
 65     // 求最短路徑
 66     void shortest_paths () {
 67         vector<path_info> path(node_count);
 68         priority_queue<node_info,vector<node_info> > min_heap;
 69         min_heap.push (node_info(0,0));    // 將起始結點入隊
 70 
 71         while (true) {
 72             node_info top = min_heap.top ();    // 取出最大值
 73             min_heap.pop ();
 74 
 75             // 已到達目的結點
 76             if (top.index == end_node) {
 77                 break ;
 78             }
 79             // 未到達則遍歷
 80             for (int i = 0; i < node_count; ++ i) {
 81                 // 頂點top.index和i間有邊,且此路徑長小於原先從原點到i的路徑長 
 82                 if (graph[top.index][i] != no_edge && 
 83                     (top.weight + graph[top.index][i]) < path[i].weight) {
 84                     min_heap.push (node_info (i,top.weight + graph[top.index][i]));
 85                     path[i].front_index = top.index;
 86                     path[i].weight = top.weight + graph[top.index][i];
 87                 }
 88             }
 89             if (min_heap.empty()) {
 90                 break ;
 91             }
 92         }
 93 
 94         shortest_path = path[end_node].weight;
 95         int index = end_node;
 96         s_path_index.push_back(index) ;
 97         while (true) {
 98             index = path[index].front_index ;
 99             s_path_index.push_back(index);
100             if (index == 0) {
101                 break;
102             }
103         }
104     }
105 
106 private:
107     vector<vector<int> >    graph ;            // 圖的數組表示
108     int                        node_count;        // 結點個數
109     const int                no_edge;        // 無通路
110     const int                end_node;        // 目的結點
111     vector<int>                s_path_index;    // 最短路徑
112     int                        shortest_path;    // 最短路徑
113 };
114 
115 int main()
116 {
117     const int size = 11; 
118     vector<vector<int> > graph (size);
119     for (int i = 0;i < size; ++ i) {
120         graph[i].resize (size);
121     }
122     for (int i = 0;i < size; ++ i) {
123         for (int j = 0;j < size; ++ j) {
124             graph[i][j] = -1;
125         }
126     }
127     graph[0][1] = 2;
128     graph[0][2] = 3;
129     graph[0][3] = 4;
130     graph[1][2] = 3;
131     graph[1][5] = 2;
132     graph[1][4] = 7;
133     graph[2][5] = 9;
134     graph[2][6] = 2;
135     graph[3][6] = 2;
136     graph[4][7] = 3;
137     graph[4][8] = 3;
138     graph[5][6] = 1;
139     graph[5][8] = 3;
140     graph[6][9] = 1;
141     graph[6][8] = 5;
142     graph[7][10] = 3;
143     graph[8][10] = 2;
144     graph[9][8] = 2;
145     graph[9][10] = 2;
146 
147     ss_shortest_paths ssp (graph, 10);
148     ssp.shortest_paths ();
149     ssp.print_spaths ();
150     return 0;
151 }
View Code

測試數據(圖)

測試結果min weight : 8path: 0 2 6 9 10
相關文章
相關標籤/搜索