alpar@74
|
1 |
// -*- mode:c++ -*-
|
alpar@74
|
2 |
|
alpar@74
|
3 |
#ifndef EDMONDS_KARP_HH
|
alpar@74
|
4 |
#define EDMONDS_KARP_HH
|
alpar@74
|
5 |
|
alpar@74
|
6 |
#include <queue>
|
alpar@74
|
7 |
|
alpar@74
|
8 |
//#include <marci_property_vector.hh>
|
alpar@74
|
9 |
|
alpar@74
|
10 |
#include <algorithm>
|
alpar@74
|
11 |
|
alpar@74
|
12 |
//#include <bfs_iterator.hh>
|
alpar@74
|
13 |
|
alpar@74
|
14 |
namespace marci {
|
alpar@74
|
15 |
template <typename Graph, typename FlowMap, typename CapacityMap>
|
alpar@74
|
16 |
typename FlowMap::ValueType maxFlow(Graph &G,
|
alpar@74
|
17 |
FlowMap &f,
|
alpar@74
|
18 |
CapacityMap &c,
|
alpar@74
|
19 |
typename Graph::NodeIt s,
|
alpar@74
|
20 |
typename Graph::NodeIt t)
|
alpar@74
|
21 |
{
|
alpar@74
|
22 |
typedef typename Graph::NodeIt NodeIt;
|
alpar@74
|
23 |
typedef typename Graph::EdgeIt EdgeIt;
|
alpar@74
|
24 |
typedef typename Graph::EachEdgeIt EachEdgeIt;
|
alpar@74
|
25 |
typedef typename Graph::OutEdgeIt OutEdgeIt;
|
alpar@74
|
26 |
typedef typename Graph::InEdgeIt InEdgeIt;
|
alpar@74
|
27 |
typedef typename FlowMap::ValueType T;
|
alpar@74
|
28 |
|
alpar@74
|
29 |
T flow_val = 0;
|
alpar@74
|
30 |
T aug_val;
|
alpar@74
|
31 |
|
alpar@74
|
32 |
for(EachEdgeIt e(G);G.valid(e);G.next(e))
|
alpar@74
|
33 |
f.set(e,0);
|
alpar@74
|
34 |
|
alpar@74
|
35 |
std::queue<NodeIt> bfs_queue;
|
alpar@74
|
36 |
typename Graph::NodeMap<int> visited(G); //0: unvisited,
|
alpar@74
|
37 |
//1: reached by a forward edge
|
alpar@74
|
38 |
//2: reached by a backward edge
|
alpar@74
|
39 |
//3: it is node s
|
alpar@74
|
40 |
typename Graph::NodeMap<EdgeIt> tree(G);
|
alpar@74
|
41 |
|
alpar@74
|
42 |
NodeIt gn; //FIXME: it might be too global for some people...
|
alpar@74
|
43 |
|
alpar@74
|
44 |
augment:
|
alpar@74
|
45 |
|
alpar@74
|
46 |
for(NodeIt n(G);G.valid(n);G.next(n))
|
alpar@74
|
47 |
visited.set(n,0);
|
alpar@74
|
48 |
|
alpar@74
|
49 |
visited.set(s,3);
|
alpar@74
|
50 |
|
alpar@74
|
51 |
//There must be a better way to do this:
|
alpar@74
|
52 |
while(!bfs_queue.empty()) bfs_queue.pop();
|
alpar@74
|
53 |
|
alpar@74
|
54 |
bfs_queue.push(s);
|
alpar@74
|
55 |
|
alpar@74
|
56 |
while(!bfs_queue.empty() && !visited.get(t))
|
alpar@74
|
57 |
{
|
alpar@74
|
58 |
NodeIt n(bfs_queue.front());
|
alpar@74
|
59 |
for(OutEdgeIt e(G,n);G.valid(e);G.next(e))
|
alpar@74
|
60 |
if(f.get(e)<c.get(e) && //FIXME: <
|
alpar@74
|
61 |
!visited.get(G.bNode(e)))
|
alpar@74
|
62 |
{
|
alpar@74
|
63 |
tree.set(G.bNode(e),e);
|
alpar@74
|
64 |
visited.set(G.bNode(e),1);
|
alpar@74
|
65 |
}
|
alpar@74
|
66 |
for(InEdgeIt e(G,n);G.valid(e);G.next(e))
|
alpar@74
|
67 |
if(f.get(e)>0 && //FIXME: >
|
alpar@74
|
68 |
!visited.get(G.bNode(e)))
|
alpar@74
|
69 |
{
|
alpar@74
|
70 |
tree.set(G.bNode(e),e);
|
alpar@74
|
71 |
visited.set(G.bNode(e),2);
|
alpar@74
|
72 |
}
|
alpar@74
|
73 |
}
|
alpar@74
|
74 |
|
alpar@74
|
75 |
if(!visited.get(t)) return flow_val;
|
alpar@74
|
76 |
|
alpar@74
|
77 |
// Augmenting value computation
|
alpar@80
|
78 |
aug_val = visited.get(t)==1 ?
|
alpar@74
|
79 |
c.get(tree.get(t))-f.get(tree.get(t)) : f.get(tree.get(t));
|
alpar@74
|
80 |
//FIXME: I would need 'G.opposite(e,n)'
|
alpar@80
|
81 |
gn = visited.get(t)==1 ? G.tail(tree.get(t)) : G.head(tree.get(t));
|
alpar@80
|
82 |
while(gn!=s) if(visited.get(gn)==1)
|
alpar@74
|
83 |
{
|
alpar@74
|
84 |
//FIXME: nonstandars. gcc extension!
|
alpar@74
|
85 |
aug_val <?= c.get(tree.get(gn))-f.get(tree.get(gn));
|
alpar@80
|
86 |
gn=G.tail(tree.get(gn));
|
alpar@74
|
87 |
}
|
alpar@74
|
88 |
else {
|
alpar@74
|
89 |
//FIXME: nonstandars. gcc extension!
|
alpar@74
|
90 |
aug_val <?= f.get(tree.get(gn));
|
alpar@80
|
91 |
gn=G.head(tree.get(gn));
|
alpar@74
|
92 |
}
|
alpar@74
|
93 |
|
alpar@74
|
94 |
// The augmentation itself
|
alpar@74
|
95 |
gn = t;
|
alpar@80
|
96 |
while(gn!=s) if(visited.get(gn)==1)
|
alpar@74
|
97 |
{
|
alpar@74
|
98 |
f.set(tree.get(gn),f.get(tree.get(gn))+aug_val);
|
alpar@80
|
99 |
gn=G.tail(tree.get(gn));
|
alpar@74
|
100 |
}
|
alpar@74
|
101 |
else {
|
alpar@74
|
102 |
f.set(tree.get(gn),f.get(tree.get(gn))-aug_val);
|
alpar@80
|
103 |
gn=G.head(tree.get(gn));
|
alpar@74
|
104 |
}
|
alpar@74
|
105 |
|
alpar@74
|
106 |
flow_val+=aug_val;
|
alpar@74
|
107 |
|
alpar@74
|
108 |
goto augment; // Vivat goto forever!
|
alpar@74
|
109 |
}
|
alpar@74
|
110 |
|
alpar@74
|
111 |
} // namespace marci
|
alpar@74
|
112 |
|
alpar@74
|
113 |
#endif //EDMONDS_KARP_HH
|