COIN-OR::LEMON - Graph Library

source: lemon-0.x/src/work/jacint/preflow.h @ 346:538ff3ce9f68

Last change on this file since 346:538ff3ce9f68 was 330:7ac0d4e8a31c, checked in by marci, 21 years ago

In the resgraphwrapper interface, and in the constructor,
the order of FlowMap? and CapacityMap? is changed.

File size: 10.4 KB
RevLine 
[109]1// -*- C++ -*-
2/*
3Heuristics:
4 2 phase
5 gap
6 list 'level_list' on the nodes on level i implemented by hand
7 stack 'active' on the active nodes on level i implemented by hand
8 runs heuristic 'highest label' for H1*n relabels
[113]9 runs heuristic 'bound decrease' for H0*n relabels, starts with 'highest label'
[109]10 
11Parameters H0 and H1 are initialized to 20 and 10.
12
[211]13Constructors:
[109]14
[211]15Preflow(Graph, Node, Node, CapMap, FlowMap)
[109]16
17Members:
18
[211]19void run()
[109]20
[211]21T flowValue() : returns the value of a maximum flow
[109]22
23void minMinCut(CutMap& M) : sets M to the characteristic vector of the
24     minimum min cut. M should be a map of bools initialized to false.
25
26void maxMinCut(CutMap& M) : sets M to the characteristic vector of the
27     maximum min cut. M should be a map of bools initialized to false.
28
29void minCut(CutMap& M) : sets M to the characteristic vector of
30     a min cut. M should be a map of bools initialized to false.
31
32*/
33
[211]34#ifndef HUGO_PREFLOW_H
35#define HUGO_PREFLOW_H
[109]36
37#define H0 20
38#define H1 1
39
40#include <vector>
41#include <queue>
42
43namespace hugo {
44
45  template <typename Graph, typename T,
[330]46            typename CapMap=typename Graph::EdgeMap<T>,
47            typename FlowMap=typename Graph::EdgeMap<T> >
[211]48  class Preflow {
[109]49   
[211]50    typedef typename Graph::Node Node;
51    typedef typename Graph::Edge Edge;
[109]52    typedef typename Graph::NodeIt NodeIt;
53    typedef typename Graph::OutEdgeIt OutEdgeIt;
54    typedef typename Graph::InEdgeIt InEdgeIt;
55   
[211]56    const Graph& G;
57    Node s;
58    Node t;
[330]59    const CapMap& capacity; 
[211]60    FlowMap& flow;
[109]61    T value;
62
63  public:
[278]64    Preflow(const Graph& _G, Node _s, Node _t, const CapMap& _capacity,
65            FlowMap& _flow ) :
[330]66      G(_G), s(_s), t(_t), capacity(_capacity), flow(_flow) {}
[211]67
68    void run() {
[109]69
70      bool phase=0;        //phase 0 is the 1st phase, phase 1 is the 2nd
71      int n=G.nodeNum();
72      int heur0=(int)(H0*n);  //time while running 'bound decrease'
73      int heur1=(int)(H1*n);  //time while running 'highest label'
74      int heur=heur1;         //starting time interval (#of relabels)
75      bool what_heur=1;       
76      /*
77        what_heur is 0 in case 'bound decrease'
78        and 1 in case 'highest label'
79      */
80      bool end=false;     
81      /*
82        Needed for 'bound decrease', 'true'
83        means no active nodes are above bound b.
84      */
85      int relabel=0;
86      int k=n-2;  //bound on the highest level under n containing a node
87      int b=k;    //bound on the highest level under n of an active node
88     
89      typename Graph::NodeMap<int> level(G,n);     
90      typename Graph::NodeMap<T> excess(G);
91
[211]92      std::vector<Node> active(n,INVALID);
93      typename Graph::NodeMap<Node> next(G,INVALID);
[109]94      //Stack of the active nodes in level i < n.
95      //We use it in both phases.
96
[211]97      typename Graph::NodeMap<Node> left(G,INVALID);
98      typename Graph::NodeMap<Node> right(G,INVALID);
99      std::vector<Node> level_list(n,INVALID);
[109]100      /*
101        List of the nodes in level i<n.
102      */
103
104      /*Reverse_bfs from t, to find the starting level.*/
105      level.set(t,0);
[211]106      std::queue<Node> bfs_queue;
[109]107      bfs_queue.push(t);
108
109      while (!bfs_queue.empty()) {
110
[211]111        Node v=bfs_queue.front();       
[109]112        bfs_queue.pop();
[211]113        int l=level[v]+1;
[109]114
[211]115        InEdgeIt e;
116        for(G.first(e,v); G.valid(e); G.next(e)) {
117          Node w=G.tail(e);
118          if ( level[w] == n && w != s ) {
[109]119            bfs_queue.push(w);
[211]120            Node first=level_list[l];
121            if ( G.valid(first) ) left.set(first,w);
[109]122            right.set(w,first);
123            level_list[l]=w;
124            level.set(w, l);
125          }
126        }
127      }
128     
129      level.set(s,n);
130     
131
132      /* Starting flow. It is everywhere 0 at the moment. */     
[211]133      OutEdgeIt e;
134      for(G.first(e,s); G.valid(e); G.next(e))
[109]135        {
[211]136          T c=capacity[e];
[109]137          if ( c == 0 ) continue;
[211]138          Node w=G.head(e);
139          if ( level[w] < n ) {   
140            if ( excess[w] == 0 && w!=t ) {
141              next.set(w,active[level[w]]);
142              active[level[w]]=w;
[109]143            }
144            flow.set(e, c);
[211]145            excess.set(w, excess[w]+c);
[109]146          }
147        }
148
149      /*
150         End of preprocessing
151      */
152
153
154
155      /*
156        Push/relabel on the highest level active nodes.
157      */       
158      while ( true ) {
159       
160        if ( b == 0 ) {
161          if ( phase ) break;
162         
163          if ( !what_heur && !end && k > 0 ) {
164            b=k;
165            end=true;
166          } else {
167            phase=1;
168            level.set(s,0);
[211]169            std::queue<Node> bfs_queue;
[109]170            bfs_queue.push(s);
171           
172            while (!bfs_queue.empty()) {
173             
[211]174              Node v=bfs_queue.front();
[109]175              bfs_queue.pop();
[211]176              int l=level[v]+1;
[109]177             
[211]178              InEdgeIt e;
179              for(G.first(e,v); G.valid(e); G.next(e)) {
180                if ( capacity[e] == flow[e] ) continue;
181                Node u=G.tail(e);
182                if ( level[u] >= n ) {
[109]183                  bfs_queue.push(u);
184                  level.set(u, l);
[211]185                  if ( excess[u] > 0 ) {
[109]186                    next.set(u,active[l]);
187                    active[l]=u;
188                  }
189                }
190              }
191           
[211]192              OutEdgeIt f;
193              for(G.first(f,v); G.valid(f); G.next(f)) {
194                if ( 0 == flow[f] ) continue;
195                Node u=G.head(f);
196                if ( level[u] >= n ) {
[109]197                  bfs_queue.push(u);
198                  level.set(u, l);
[211]199                  if ( excess[u] > 0 ) {
[109]200                    next.set(u,active[l]);
201                    active[l]=u;
202                  }
203                }
204              }
205            }
206            b=n-2;
207            }
208           
209        }
210         
211         
[211]212        if ( !G.valid(active[b]) ) --b;
[109]213        else {
214          end=false; 
215
[211]216          Node w=active[b];
217          active[b]=next[w];
218          int lev=level[w];
219          T exc=excess[w];
[109]220          int newlevel=n;       //bound on the next level of w
221         
[211]222          OutEdgeIt e;
223          for(G.first(e,w); G.valid(e); G.next(e)) {
[109]224           
[211]225            if ( flow[e] == capacity[e] ) continue;
226            Node v=G.head(e);           
[109]227            //e=wv         
228           
[211]229            if( lev > level[v] ) {     
[109]230              /*Push is allowed now*/
231             
[211]232              if ( excess[v]==0 && v!=t && v!=s ) {
233                int lev_v=level[v];
[109]234                next.set(v,active[lev_v]);
235                active[lev_v]=v;
236              }
237             
[211]238              T cap=capacity[e];
239              T flo=flow[e];
[109]240              T remcap=cap-flo;
241             
242              if ( remcap >= exc ) {       
243                /*A nonsaturating push.*/
244               
245                flow.set(e, flo+exc);
[211]246                excess.set(v, excess[v]+exc);
[109]247                exc=0;
248                break;
249               
250              } else {
251                /*A saturating push.*/
252               
253                flow.set(e, cap);
[211]254                excess.set(v, excess[v]+remcap);
[109]255                exc-=remcap;
256              }
[211]257            } else if ( newlevel > level[v] ){
258              newlevel = level[v];
[109]259            }       
260           
261          } //for out edges wv
262       
263       
264        if ( exc > 0 ) {       
[211]265          InEdgeIt e;
266          for(G.first(e,w); G.valid(e); G.next(e)) {
[109]267           
[211]268            if( flow[e] == 0 ) continue;
269            Node v=G.tail(e); 
[109]270            //e=vw
271           
[211]272            if( lev > level[v] ) { 
[109]273              /*Push is allowed now*/
274             
[211]275              if ( excess[v]==0 && v!=t && v!=s ) {
276                int lev_v=level[v];
[109]277                next.set(v,active[lev_v]);
278                active[lev_v]=v;
279              }
280             
[211]281              T flo=flow[e];
[109]282             
283              if ( flo >= exc ) {
284                /*A nonsaturating push.*/
285               
286                flow.set(e, flo-exc);
[211]287                excess.set(v, excess[v]+exc);
[109]288                exc=0;
289                break;
290              } else {                                               
291                /*A saturating push.*/
292               
[211]293                excess.set(v, excess[v]+flo);
[109]294                exc-=flo;
295                flow.set(e,0);
296              } 
[211]297            } else if ( newlevel > level[v] ) {
298              newlevel = level[v];
[109]299            }       
300          } //for in edges vw
301         
302        } // if w still has excess after the out edge for cycle
303       
304        excess.set(w, exc);
305         
306        /*
307          Relabel
308        */
309       
310
311        if ( exc > 0 ) {
312          //now 'lev' is the old level of w
313       
314          if ( phase ) {
315            level.set(w,++newlevel);
316            next.set(w,active[newlevel]);
317            active[newlevel]=w;
318            b=newlevel;
319          } else {
320            //unlacing starts
[211]321            Node right_n=right[w];
322            Node left_n=left[w];
[109]323
[211]324            if ( G.valid(right_n) ) {
325              if ( G.valid(left_n) ) {
[109]326                right.set(left_n, right_n);
327                left.set(right_n, left_n);
328              } else {
329                level_list[lev]=right_n;   
[211]330                left.set(right_n, INVALID);
[109]331              }
332            } else {
[211]333              if ( G.valid(left_n) ) {
334                right.set(left_n, INVALID);
[109]335              } else {
[211]336                level_list[lev]=INVALID;   
[109]337              }
338            }
339            //unlacing ends
340               
341            //gapping starts
[211]342            if ( !G.valid(level_list[lev]) ) {
[109]343             
344              for (int i=lev; i!=k ; ) {
[211]345                Node v=level_list[++i];
346                while ( G.valid(v) ) {
[109]347                  level.set(v,n);
[211]348                  v=right[v];
[109]349                }
[211]350                level_list[i]=INVALID;
351                if ( !what_heur ) active[i]=INVALID;
[109]352              }     
353
354              level.set(w,n);
355              b=lev-1;
356              k=b;
357              //gapping ends
358            } else {
359             
360              if ( newlevel == n ) level.set(w,n);
361              else {
362                level.set(w,++newlevel);
363                next.set(w,active[newlevel]);
364                active[newlevel]=w;
365                if ( what_heur ) b=newlevel;
366                if ( k < newlevel ) ++k;
[211]367                Node first=level_list[newlevel];
368                if ( G.valid(first) ) left.set(first,w);
[109]369                right.set(w,first);
[211]370                left.set(w,INVALID);
[109]371                level_list[newlevel]=w;
372              }
373            }
374
375
376            ++relabel;
377            if ( relabel >= heur ) {
378              relabel=0;
379              if ( what_heur ) {
380                what_heur=0;
381                heur=heur0;
382                end=false;
383              } else {
384                what_heur=1;
385                heur=heur1;
386                b=k;
387              }
388            }
389          } //phase 0
390         
391         
392        } // if ( exc > 0 )
393         
394       
395        }  // if stack[b] is nonempty
396       
397      } // while(true)
398
399
[211]400      value = excess[t];
[109]401      /*Max flow value.*/
402     
403    } //void run()
404
405
406
407
408
409    /*
410      Returns the maximum value of a flow.
411     */
412
[211]413    T flowValue() {
[109]414      return value;
415    }
416
417
418    FlowMap Flow() {
419      return flow;
420      }
421
422
423   
424    void Flow(FlowMap& _flow ) {
[211]425      NodeIt v;
426      for(G.first(v) ; G.valid(v); G.next(v))
427        _flow.set(v,flow[v]);
428    }
[109]429
430
431
432    /*
433      Returns the minimum min cut, by a bfs from s in the residual graph.
434    */
435   
436    template<typename _CutMap>
437    void minMinCut(_CutMap& M) {
438   
[211]439      std::queue<Node> queue;
[109]440     
441      M.set(s,true);     
442      queue.push(s);
443
444      while (!queue.empty()) {
[211]445        Node w=queue.front();
[109]446        queue.pop();
447
[211]448        OutEdgeIt e;
449        for(G.first(e,w) ; G.valid(e); G.next(e)) {
450          Node v=G.head(e);
451          if (!M[v] && flow[e] < capacity[e] ) {
[109]452            queue.push(v);
453            M.set(v, true);
454          }
455        }
456
[211]457        InEdgeIt f;
458        for(G.first(f,w) ; G.valid(f); G.next(f)) {
459          Node v=G.tail(f);
460          if (!M[v] && flow[f] > 0 ) {
[109]461            queue.push(v);
462            M.set(v, true);
463          }
464        }
465      }
466    }
467
468
469 
470    /*
471      Returns the maximum min cut, by a reverse bfs
472      from t in the residual graph.
473    */
474   
475    template<typename _CutMap>
476    void maxMinCut(_CutMap& M) {
477   
[211]478      std::queue<Node> queue;
[109]479     
480      M.set(t,true);       
481      queue.push(t);
482
483      while (!queue.empty()) {
[211]484        Node w=queue.front();
[109]485        queue.pop();
486
[211]487
488        InEdgeIt e;
489        for(G.first(e,w) ; G.valid(e); G.next(e)) {
490          Node v=G.tail(e);
491          if (!M[v] && flow[e] < capacity[e] ) {
[109]492            queue.push(v);
493            M.set(v, true);
494          }
495        }
[211]496       
497        OutEdgeIt f;
498        for(G.first(f,w) ; G.valid(f); G.next(f)) {
499          Node v=G.head(f);
500          if (!M[v] && flow[f] > 0 ) {
[109]501            queue.push(v);
502            M.set(v, true);
503          }
504        }
505      }
506
[211]507      NodeIt v;
508      for(G.first(v) ; G.valid(v); G.next(v)) {
509        M.set(v, !M[v]);
[109]510      }
511
512    }
513
514
515
516    template<typename CutMap>
517    void minCut(CutMap& M) {
518      minMinCut(M);
519    }
520
521
522  };
523
[174]524} //namespace hugo
[109]525
[174]526#endif //PREFLOW_H
[109]527
528
[174]529
530
Note: See TracBrowser for help on using the repository browser.