jacint@1057
|
1 |
// -*- C++ -*-
|
jacint@1057
|
2 |
#ifndef LEMON_MAX_MATCHING_H
|
jacint@1057
|
3 |
#define LEMON_MAX_MATCHING_H
|
jacint@1057
|
4 |
|
jacint@1057
|
5 |
///\ingroup galgs
|
jacint@1057
|
6 |
///\file
|
jacint@1057
|
7 |
///\brief Maximum matching algorithm.
|
jacint@1057
|
8 |
|
jacint@1057
|
9 |
#include <queue>
|
jacint@1057
|
10 |
|
jacint@1057
|
11 |
|
jacint@1057
|
12 |
#include <iostream>
|
jacint@1057
|
13 |
|
jacint@1057
|
14 |
|
jacint@1057
|
15 |
|
jacint@1057
|
16 |
#include <invalid.h>
|
jacint@1057
|
17 |
#include <unionfind.h>
|
jacint@1057
|
18 |
#include <lemon/graph_utils.h>
|
jacint@1057
|
19 |
|
jacint@1057
|
20 |
namespace lemon {
|
jacint@1057
|
21 |
|
jacint@1057
|
22 |
/// \addtogroup galgs
|
jacint@1057
|
23 |
/// @{
|
jacint@1057
|
24 |
|
jacint@1057
|
25 |
///Maximum matching algorithms class.
|
jacint@1057
|
26 |
|
jacint@1057
|
27 |
///This class provides Edmonds' alternating forest matching
|
jacint@1057
|
28 |
///algorithm. The starting matching (if any) can be passed to the
|
jacint@1057
|
29 |
///algorithm using read-in functions \ref readNMapNode, \ref
|
jacint@1057
|
30 |
///readNMapEdge or \ref readEMapBool depending on the container. The
|
jacint@1057
|
31 |
///resulting maximum matching can be attained by write-out functions
|
jacint@1057
|
32 |
///\ref writeNMapNode, \ref writeNMapEdge or \ref writeEMapBool
|
jacint@1057
|
33 |
///depending on the preferred container.
|
jacint@1057
|
34 |
///
|
jacint@1057
|
35 |
///The dual side of a mathcing is a map of the nodes to
|
jacint@1057
|
36 |
///MaxMatching::pos_enum, having values D, A and C showing the
|
jacint@1057
|
37 |
///Gallai-Edmonds decomposition of the graph. The nodes in D induce
|
jacint@1057
|
38 |
///a graph with factor-critical components, the nodes in A form the
|
jacint@1057
|
39 |
///barrier, and the nodes in C induce a graph having a perfect
|
jacint@1057
|
40 |
///matching. This decomposition can be attained by calling \ref
|
jacint@1057
|
41 |
///writePos after running the algorithm. Before subsequent runs,
|
jacint@1057
|
42 |
///the function \ref resetPos() must be called.
|
jacint@1057
|
43 |
///
|
jacint@1057
|
44 |
///\param Graph The undirected graph type the algorithm runs on.
|
jacint@1057
|
45 |
///
|
jacint@1057
|
46 |
///\author Jacint Szabo
|
jacint@1057
|
47 |
template <typename Graph>
|
jacint@1057
|
48 |
class MaxMatching {
|
jacint@1057
|
49 |
typedef typename Graph::Node Node;
|
jacint@1057
|
50 |
typedef typename Graph::Edge Edge;
|
jacint@1057
|
51 |
typedef typename Graph::UndirEdgeIt UndirEdgeIt;
|
jacint@1057
|
52 |
typedef typename Graph::NodeIt NodeIt;
|
jacint@1057
|
53 |
typedef typename Graph::IncEdgeIt IncEdgeIt;
|
jacint@1057
|
54 |
|
jacint@1057
|
55 |
typedef UnionFindEnum<Node, Graph::template NodeMap> UFE;
|
jacint@1057
|
56 |
|
jacint@1057
|
57 |
public:
|
jacint@1057
|
58 |
|
jacint@1057
|
59 |
///Indicates the Gallai-Edmonds decomposition of the graph.
|
jacint@1057
|
60 |
|
jacint@1057
|
61 |
///Indicates the Gallai-Edmonds decomposition of the graph, which
|
jacint@1057
|
62 |
///shows an upper bound on the size of a maximum matching. The
|
jacint@1057
|
63 |
///nodes with pos_enum \c D induce a graph with factor-critical
|
jacint@1057
|
64 |
///components, the nodes in \c A form the canonical barrier, and the
|
jacint@1057
|
65 |
///nodes in \c C induce a graph having a perfect matching.
|
jacint@1057
|
66 |
enum pos_enum {
|
jacint@1057
|
67 |
D=0,
|
jacint@1057
|
68 |
A=1,
|
jacint@1057
|
69 |
C=2
|
jacint@1057
|
70 |
};
|
jacint@1057
|
71 |
|
jacint@1057
|
72 |
private:
|
jacint@1057
|
73 |
|
jacint@1057
|
74 |
static const int HEUR_density=2;
|
jacint@1057
|
75 |
const Graph& g;
|
jacint@1057
|
76 |
typename Graph::template NodeMap<Node> mate;
|
jacint@1057
|
77 |
typename Graph::template NodeMap<pos_enum> position;
|
jacint@1057
|
78 |
|
jacint@1057
|
79 |
public:
|
jacint@1057
|
80 |
|
jacint@1057
|
81 |
MaxMatching(const Graph& _g) : g(_g), mate(_g,INVALID), position(_g,C) {}
|
jacint@1057
|
82 |
|
jacint@1057
|
83 |
///Runs Edmonds' algorithm.
|
jacint@1057
|
84 |
|
jacint@1057
|
85 |
///Runs Edmonds' algorithm for sparse graphs (countEdges <=
|
jacint@1057
|
86 |
///2*countNodes), and a heuristical Edmonds' algorithm with a
|
jacint@1057
|
87 |
///heuristic of postponing shrinks for dense graphs. \pre Before
|
jacint@1057
|
88 |
///the subsequent calls \ref resetPos must be called.
|
jacint@1057
|
89 |
inline void run();
|
jacint@1057
|
90 |
|
jacint@1057
|
91 |
///Runs Edmonds' algorithm.
|
jacint@1057
|
92 |
|
jacint@1057
|
93 |
///If heur=0 it runs Edmonds' algorithm. If heur=1 it runs
|
jacint@1057
|
94 |
///Edmonds' algorithm with a heuristic of postponing shrinks,
|
jacint@1057
|
95 |
///giving a faster algorithm for dense graphs. \pre Before the
|
jacint@1057
|
96 |
///subsequent calls \ref resetPos must be called.
|
jacint@1057
|
97 |
void runEdmonds( int heur );
|
jacint@1057
|
98 |
|
jacint@1057
|
99 |
///Finds a greedy matching starting from the actual matching.
|
jacint@1057
|
100 |
|
jacint@1057
|
101 |
///Starting form the actual matching stored, it finds a maximal
|
jacint@1057
|
102 |
///greedy matching.
|
jacint@1057
|
103 |
void greedyMatching();
|
jacint@1057
|
104 |
|
jacint@1057
|
105 |
///Returns the size of the actual matching stored.
|
jacint@1057
|
106 |
|
jacint@1057
|
107 |
///Returns the size of the actual matching stored. After \ref
|
jacint@1057
|
108 |
///run() it returns the size of a maximum matching in the graph.
|
jacint@1057
|
109 |
int size () const;
|
jacint@1057
|
110 |
|
jacint@1057
|
111 |
///Resets the map storing the Gallai-Edmonds decomposition.
|
jacint@1057
|
112 |
|
jacint@1057
|
113 |
///Resets the map storing the Gallai-Edmonds decomposition of the
|
jacint@1057
|
114 |
///graph, making it possible to run the algorithm. Must be called
|
jacint@1057
|
115 |
///before all runs of the Edmonds algorithm, except for the first
|
jacint@1057
|
116 |
///run.
|
jacint@1057
|
117 |
void resetPos();
|
jacint@1057
|
118 |
|
jacint@1057
|
119 |
///Resets the actual matching to the empty matching.
|
jacint@1057
|
120 |
|
jacint@1057
|
121 |
///Resets the actual matching to the empty matching.
|
jacint@1057
|
122 |
///
|
jacint@1057
|
123 |
void resetMatching();
|
jacint@1057
|
124 |
|
jacint@1057
|
125 |
///Reads a matching from a \c Node map of \c Nodes.
|
jacint@1057
|
126 |
|
jacint@1057
|
127 |
///Reads a matching from a \c Node map of \c Nodes. This map must be \e
|
jacint@1057
|
128 |
///symmetric, i.e. if \c map[u]=v then \c map[v]=u must hold, and
|
jacint@1057
|
129 |
///\c uv will be an edge of the matching.
|
jacint@1057
|
130 |
template<typename NMapN>
|
jacint@1057
|
131 |
void readNMapNode(NMapN& map) {
|
jacint@1057
|
132 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
133 |
mate.set(v,map[v]);
|
jacint@1057
|
134 |
}
|
jacint@1057
|
135 |
}
|
jacint@1057
|
136 |
|
jacint@1057
|
137 |
///Writes the stored matching to a \c Node map of \c Nodes.
|
jacint@1057
|
138 |
|
jacint@1057
|
139 |
///Writes the stored matching to a \c Node map of \c Nodes. The
|
jacint@1057
|
140 |
///resulting map will be \e symmetric, i.e. if \c map[u]=v then \c
|
jacint@1057
|
141 |
///map[v]=u will hold, and now \c uv is an edge of the matching.
|
jacint@1057
|
142 |
template<typename NMapN>
|
jacint@1057
|
143 |
void writeNMapNode (NMapN& map) const {
|
jacint@1057
|
144 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
145 |
map.set(v,mate[v]);
|
jacint@1057
|
146 |
}
|
jacint@1057
|
147 |
}
|
jacint@1057
|
148 |
|
jacint@1057
|
149 |
///Reads a matching from a \c Node map of \c Edges.
|
jacint@1057
|
150 |
|
jacint@1057
|
151 |
///Reads a matching from a \c Node map of incident \c Edges. This
|
jacint@1057
|
152 |
///map must have the property that if \c G.target(map[u])=v then \c
|
jacint@1057
|
153 |
///G.target(map[v])=u must hold, and now this edge is an edge of
|
jacint@1057
|
154 |
///the matching.
|
jacint@1057
|
155 |
template<typename NMapE>
|
jacint@1057
|
156 |
void readNMapEdge(NMapE& map) {
|
jacint@1057
|
157 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
158 |
Edge e=map[v];
|
jacint@1057
|
159 |
if ( g.valid(e) )
|
jacint@1057
|
160 |
g.source(e) == v ? mate.set(v,g.target(e)) : mate.set(v,g.source(e));
|
jacint@1057
|
161 |
}
|
jacint@1057
|
162 |
}
|
jacint@1057
|
163 |
|
jacint@1057
|
164 |
///Writes the matching stored to a \c Node map of \c Edges.
|
jacint@1057
|
165 |
|
jacint@1057
|
166 |
///Writes the stored matching to a \c Node map of incident \c
|
jacint@1057
|
167 |
///Edges. This map will have the property that if \c
|
jacint@1057
|
168 |
///g.target(map[u])=v then \c g.target(map[v])=u holds, and now this
|
jacint@1057
|
169 |
///edge is an edge of the matching.
|
jacint@1057
|
170 |
template<typename NMapE>
|
jacint@1057
|
171 |
void writeNMapEdge (NMapE& map) const {
|
jacint@1057
|
172 |
typename Graph::template NodeMap<bool> todo(g,true);
|
jacint@1057
|
173 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
174 |
if ( todo[v] && mate[v]!=INVALID ) {
|
jacint@1057
|
175 |
Node u=mate[v];
|
jacint@1057
|
176 |
for(IncEdgeIt e(g,v); e!=INVALID; ++e) {
|
jacint@1057
|
177 |
if ( g.target(e) == u ) {
|
jacint@1057
|
178 |
map.set(u,e);
|
jacint@1057
|
179 |
map.set(v,e);
|
jacint@1057
|
180 |
todo.set(u,false);
|
jacint@1057
|
181 |
todo.set(v,false);
|
jacint@1057
|
182 |
break;
|
jacint@1057
|
183 |
}
|
jacint@1057
|
184 |
}
|
jacint@1057
|
185 |
}
|
jacint@1057
|
186 |
}
|
jacint@1057
|
187 |
}
|
jacint@1057
|
188 |
|
jacint@1057
|
189 |
|
jacint@1057
|
190 |
///Reads a matching from an \c Edge map of \c bools.
|
jacint@1057
|
191 |
|
jacint@1057
|
192 |
///Reads a matching from an \c Edge map of \c bools. This map must
|
jacint@1057
|
193 |
///have the property that there are no two adjacent edges \c e, \c
|
jacint@1057
|
194 |
///f with \c map[e]=map[f]=true. The edges \c e with \c
|
jacint@1057
|
195 |
///map[e]=true form the matching.
|
jacint@1057
|
196 |
template<typename EMapB>
|
jacint@1057
|
197 |
void readEMapBool(EMapB& map) {
|
jacint@1057
|
198 |
for(UndirEdgeIt e(g); e!=INVALID; ++e) {
|
jacint@1057
|
199 |
if ( map[e] ) {
|
jacint@1057
|
200 |
Node u=g.source(e);
|
jacint@1057
|
201 |
Node v=g.target(e);
|
jacint@1057
|
202 |
mate.set(u,v);
|
jacint@1057
|
203 |
mate.set(v,u);
|
jacint@1057
|
204 |
}
|
jacint@1057
|
205 |
}
|
jacint@1057
|
206 |
}
|
jacint@1057
|
207 |
//iterable boolmap?
|
jacint@1057
|
208 |
|
jacint@1057
|
209 |
|
jacint@1057
|
210 |
///Writes the matching stored to an \c Edge map of \c bools.
|
jacint@1057
|
211 |
|
jacint@1057
|
212 |
///Writes the matching stored to an \c Edge map of \c bools. This
|
jacint@1057
|
213 |
///map will have the property that there are no two adjacent edges
|
jacint@1057
|
214 |
///\c e, \c f with \c map[e]=map[f]=true. The edges \c e with \c
|
jacint@1057
|
215 |
///map[e]=true form the matching.
|
jacint@1057
|
216 |
template<typename EMapB>
|
jacint@1057
|
217 |
void writeEMapBool (EMapB& map) const {
|
jacint@1057
|
218 |
for(UndirEdgeIt e(g); e!=INVALID; ++e) map.set(e,false);
|
jacint@1057
|
219 |
|
jacint@1057
|
220 |
typename Graph::template NodeMap<bool> todo(g,true);
|
jacint@1057
|
221 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
222 |
if ( todo[v] && mate[v]!=INVALID ) {
|
jacint@1057
|
223 |
Node u=mate[v];
|
jacint@1057
|
224 |
for(IncEdgeIt e(g,v); e!=INVALID; ++e) {
|
jacint@1057
|
225 |
if ( g.target(e) == u ) {
|
jacint@1057
|
226 |
map.set(e,true);
|
jacint@1057
|
227 |
todo.set(u,false);
|
jacint@1057
|
228 |
todo.set(v,false);
|
jacint@1057
|
229 |
break;
|
jacint@1057
|
230 |
}
|
jacint@1057
|
231 |
}
|
jacint@1057
|
232 |
}
|
jacint@1057
|
233 |
}
|
jacint@1057
|
234 |
}
|
jacint@1057
|
235 |
|
jacint@1057
|
236 |
|
jacint@1057
|
237 |
///Writes the canonical decomposition of the graph after running
|
jacint@1057
|
238 |
///the algorithm.
|
jacint@1057
|
239 |
|
jacint@1057
|
240 |
///After calling any run methods of the class, and before calling
|
jacint@1057
|
241 |
///\ref resetPos(), it writes the Gallai-Edmonds canonical
|
jacint@1057
|
242 |
///decomposition of the graph. \c map must be a node map
|
jacint@1057
|
243 |
///of \ref pos_enum 's.
|
jacint@1057
|
244 |
template<typename NMapEnum>
|
jacint@1057
|
245 |
void writePos (NMapEnum& map) const {
|
jacint@1057
|
246 |
for(NodeIt v(g); v!=INVALID; ++v) map.set(v,position[v]);
|
jacint@1057
|
247 |
}
|
jacint@1057
|
248 |
|
jacint@1057
|
249 |
private:
|
jacint@1057
|
250 |
|
jacint@1057
|
251 |
void lateShrink(Node v, typename Graph::template NodeMap<Node>& ear,
|
jacint@1057
|
252 |
UFE& blossom, UFE& tree);
|
jacint@1057
|
253 |
|
jacint@1057
|
254 |
void normShrink(Node v, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
255 |
UFE& blossom, UFE& tree);
|
jacint@1057
|
256 |
|
jacint@1057
|
257 |
bool noShrinkStep(Node x, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
258 |
UFE& blossom, UFE& tree, std::queue<Node>& Q);
|
jacint@1057
|
259 |
|
jacint@1057
|
260 |
void shrinkStep(Node& top, Node& middle, Node& bottom, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
261 |
UFE& blossom, UFE& tree, std::queue<Node>& Q);
|
jacint@1057
|
262 |
|
jacint@1057
|
263 |
void augment(Node x, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
264 |
UFE& blossom, UFE& tree);
|
jacint@1057
|
265 |
};
|
jacint@1057
|
266 |
|
jacint@1057
|
267 |
|
jacint@1057
|
268 |
// **********************************************************************
|
jacint@1057
|
269 |
// IMPLEMENTATIONS
|
jacint@1057
|
270 |
// **********************************************************************
|
jacint@1057
|
271 |
|
jacint@1057
|
272 |
|
jacint@1057
|
273 |
template <typename Graph>
|
jacint@1057
|
274 |
void MaxMatching<Graph>::run() {
|
jacint@1057
|
275 |
if ( countUndirEdges(g) < HEUR_density*countNodes(g) ) {
|
jacint@1057
|
276 |
greedyMatching();
|
jacint@1057
|
277 |
runEdmonds(1);
|
jacint@1057
|
278 |
} else runEdmonds(0);
|
jacint@1057
|
279 |
}
|
jacint@1057
|
280 |
|
jacint@1057
|
281 |
|
jacint@1057
|
282 |
template <typename Graph>
|
jacint@1057
|
283 |
void MaxMatching<Graph>::runEdmonds( int heur=1 ) {
|
jacint@1057
|
284 |
|
jacint@1057
|
285 |
std::cout<<"Entering runEdmonds"<<std::endl;
|
jacint@1057
|
286 |
|
jacint@1057
|
287 |
typename Graph::template NodeMap<Node> ear(g,INVALID);
|
jacint@1057
|
288 |
//undefined for the base nodes of the blossoms (i.e. for the
|
jacint@1057
|
289 |
//representative elements of UFE blossom) and for the nodes in C
|
jacint@1057
|
290 |
|
jacint@1057
|
291 |
typename UFE::MapType blossom_base(g);
|
jacint@1057
|
292 |
UFE blossom(blossom_base);
|
jacint@1057
|
293 |
typename UFE::MapType tree_base(g);
|
jacint@1057
|
294 |
UFE tree(tree_base);
|
jacint@1057
|
295 |
|
jacint@1057
|
296 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
297 |
if ( position[v]==C && mate[v]==INVALID ) {
|
jacint@1057
|
298 |
blossom.insert(v);
|
jacint@1057
|
299 |
tree.insert(v);
|
jacint@1057
|
300 |
position.set(v,D);
|
jacint@1057
|
301 |
if ( heur == 1 ) lateShrink( v, ear, blossom, tree );
|
jacint@1057
|
302 |
else normShrink( v, ear, blossom, tree );
|
jacint@1057
|
303 |
}
|
jacint@1057
|
304 |
}
|
jacint@1057
|
305 |
|
jacint@1057
|
306 |
|
jacint@1057
|
307 |
std::cout<<" runEdmonds end"<<std::endl;
|
jacint@1057
|
308 |
|
jacint@1057
|
309 |
|
jacint@1057
|
310 |
}
|
jacint@1057
|
311 |
|
jacint@1057
|
312 |
template <typename Graph>
|
jacint@1057
|
313 |
void MaxMatching<Graph>::lateShrink(Node v, typename Graph::template NodeMap<Node>& ear,
|
jacint@1057
|
314 |
UFE& blossom, UFE& tree) {
|
jacint@1057
|
315 |
|
jacint@1057
|
316 |
|
jacint@1057
|
317 |
std::cout<<"Entering lateShrink"<<std::endl;
|
jacint@1057
|
318 |
|
jacint@1057
|
319 |
|
jacint@1057
|
320 |
std::queue<Node> Q; //queue of the totally unscanned nodes
|
jacint@1057
|
321 |
Q.push(v);
|
jacint@1057
|
322 |
std::queue<Node> R;
|
jacint@1057
|
323 |
//queue of the nodes which must be scanned for a possible shrink
|
jacint@1057
|
324 |
|
jacint@1057
|
325 |
while ( !Q.empty() ) {
|
jacint@1057
|
326 |
Node x=Q.front();
|
jacint@1057
|
327 |
Q.pop();
|
jacint@1057
|
328 |
if ( noShrinkStep( x, ear, blossom, tree, Q ) ) return;
|
jacint@1057
|
329 |
else R.push(x);
|
jacint@1057
|
330 |
}
|
jacint@1057
|
331 |
|
jacint@1057
|
332 |
while ( !R.empty() ) {
|
jacint@1057
|
333 |
Node x=R.front();
|
jacint@1057
|
334 |
R.pop();
|
jacint@1057
|
335 |
|
jacint@1057
|
336 |
for( IncEdgeIt e(g,x); e!=INVALID ; ++e ) {
|
jacint@1057
|
337 |
Node y=g.target(e);
|
jacint@1057
|
338 |
|
jacint@1057
|
339 |
if ( position[y] == D && blossom.find(x) != blossom.find(y) ) {
|
jacint@1057
|
340 |
//x and y must be in the same tree//biztos? az oddbol d-belive lettek is?
|
jacint@1057
|
341 |
|
jacint@1057
|
342 |
typename Graph::template NodeMap<bool> path(g,false);
|
jacint@1057
|
343 |
|
jacint@1057
|
344 |
Node b=blossom.find(x);
|
jacint@1057
|
345 |
path.set(b,true);
|
jacint@1057
|
346 |
b=mate[b];
|
jacint@1057
|
347 |
while ( b!=INVALID ) {
|
jacint@1057
|
348 |
b=blossom.find(ear[b]);
|
jacint@1057
|
349 |
path.set(b,true);
|
jacint@1057
|
350 |
b=mate[b];
|
jacint@1057
|
351 |
} //going till the root
|
jacint@1057
|
352 |
|
jacint@1057
|
353 |
Node top=y;
|
jacint@1057
|
354 |
Node middle=blossom.find(top);
|
jacint@1057
|
355 |
Node bottom=x;
|
jacint@1057
|
356 |
while ( !path[middle] )
|
jacint@1057
|
357 |
shrinkStep(top, middle, bottom, ear, blossom, tree, Q);
|
jacint@1057
|
358 |
|
jacint@1057
|
359 |
Node base=middle;
|
jacint@1057
|
360 |
top=x;
|
jacint@1057
|
361 |
middle=blossom.find(top);
|
jacint@1057
|
362 |
bottom=y;
|
jacint@1057
|
363 |
Node blossom_base=blossom.find(base);
|
jacint@1057
|
364 |
while ( middle!=blossom_base )
|
jacint@1057
|
365 |
shrinkStep(top, middle, bottom, ear, blossom, tree, Q);
|
jacint@1057
|
366 |
|
jacint@1057
|
367 |
blossom.makeRep(base);
|
jacint@1057
|
368 |
} // if shrink is needed
|
jacint@1057
|
369 |
|
jacint@1057
|
370 |
//most nehany odd node is d-beli lett, es rajuk az is megnezendo hogy mely d-beliekkel szonszedosak mas faban
|
jacint@1057
|
371 |
|
jacint@1057
|
372 |
while ( !Q.empty() ) {
|
jacint@1057
|
373 |
Node x=Q.front();
|
jacint@1057
|
374 |
Q.pop();
|
jacint@1057
|
375 |
if ( noShrinkStep(x, ear, blossom, tree, Q) ) return;
|
jacint@1057
|
376 |
else R.push(x);
|
jacint@1057
|
377 |
}
|
jacint@1057
|
378 |
} //for e
|
jacint@1057
|
379 |
} // while ( !R.empty() )
|
jacint@1057
|
380 |
}
|
jacint@1057
|
381 |
|
jacint@1057
|
382 |
|
jacint@1057
|
383 |
template <typename Graph>
|
jacint@1057
|
384 |
void MaxMatching<Graph>::normShrink(Node v, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
385 |
UFE& blossom, UFE& tree) {
|
jacint@1057
|
386 |
|
jacint@1057
|
387 |
|
jacint@1057
|
388 |
std::cout<<"Entering normShrink with node "<<g.id(v)<<std::endl;
|
jacint@1057
|
389 |
|
jacint@1057
|
390 |
|
jacint@1057
|
391 |
std::queue<Node> Q; //queue of the unscanned nodes
|
jacint@1057
|
392 |
Q.push(v);
|
jacint@1057
|
393 |
while ( !Q.empty() ) {
|
jacint@1057
|
394 |
|
jacint@1057
|
395 |
std::cout<<"beginning of norm while"<<std::endl;
|
jacint@1057
|
396 |
|
jacint@1057
|
397 |
Node x=Q.front();
|
jacint@1057
|
398 |
Q.pop();
|
jacint@1057
|
399 |
|
jacint@1057
|
400 |
for( IncEdgeIt e(g,x); e!=INVALID; ++e ) {
|
jacint@1057
|
401 |
|
jacint@1057
|
402 |
|
jacint@1057
|
403 |
for( IncEdgeIt f(g,x); f!=INVALID; ++f ) {
|
jacint@1057
|
404 |
std::cout<<"Starting for." <<std::endl;
|
jacint@1057
|
405 |
std::cout<<"edges " << g.id(f)<< " : " << g.id(g.target(f))<<std::endl;
|
jacint@1057
|
406 |
std::cout<<"Ending for." <<std::endl;
|
jacint@1057
|
407 |
}
|
jacint@1057
|
408 |
|
jacint@1057
|
409 |
std::cout<<"Ending the whole for." <<std::endl;
|
jacint@1057
|
410 |
std::cout<<"for (In normShrink) with edge " << g.id(e)<< " : " << g.id(x);
|
jacint@1057
|
411 |
|
jacint@1057
|
412 |
Node y=g.target(e);
|
jacint@1057
|
413 |
|
jacint@1057
|
414 |
std::cout<<" "<<g.id(y)<<std::endl;
|
jacint@1057
|
415 |
|
jacint@1057
|
416 |
switch ( position[y] ) {
|
jacint@1057
|
417 |
case D: //x and y must be in the same tree //asszem nem!!!
|
jacint@1057
|
418 |
|
jacint@1057
|
419 |
std::cout<<" pos[y] " << position[y]<<std::endl;
|
jacint@1057
|
420 |
std::cout<<" blossom.find(x) ="<< g.id(blossom.find(x))<<std::endl;
|
jacint@1057
|
421 |
std::cout<<" blossom.find(y) ="<< g.id(blossom.find(y))<<std::endl;
|
jacint@1057
|
422 |
|
jacint@1057
|
423 |
|
jacint@1057
|
424 |
if ( blossom.find(x) != blossom.find(y) ) { //shrink
|
jacint@1057
|
425 |
typename Graph::template NodeMap<bool> path(g,false);
|
jacint@1057
|
426 |
|
jacint@1057
|
427 |
Node b=blossom.find(x);
|
jacint@1057
|
428 |
path.set(b,true);
|
jacint@1057
|
429 |
b=mate[b];
|
jacint@1057
|
430 |
while ( b!=INVALID ) {
|
jacint@1057
|
431 |
b=blossom.find(ear[b]);
|
jacint@1057
|
432 |
path.set(b,true);
|
jacint@1057
|
433 |
b=mate[b];
|
jacint@1057
|
434 |
} //going till the root
|
jacint@1057
|
435 |
|
jacint@1057
|
436 |
Node top=y;
|
jacint@1057
|
437 |
Node middle=blossom.find(top);
|
jacint@1057
|
438 |
Node bottom=x;
|
jacint@1057
|
439 |
while ( !path[middle] )
|
jacint@1057
|
440 |
shrinkStep(top, middle, bottom, ear, blossom, tree, Q);
|
jacint@1057
|
441 |
|
jacint@1057
|
442 |
Node base=middle;
|
jacint@1057
|
443 |
top=x;
|
jacint@1057
|
444 |
middle=blossom.find(top);
|
jacint@1057
|
445 |
bottom=y;
|
jacint@1057
|
446 |
Node blossom_base=blossom.find(base);
|
jacint@1057
|
447 |
while ( middle!=blossom_base )
|
jacint@1057
|
448 |
shrinkStep(top, middle, bottom, ear, blossom, tree, Q);
|
jacint@1057
|
449 |
|
jacint@1057
|
450 |
blossom.makeRep(base);
|
jacint@1057
|
451 |
}
|
jacint@1057
|
452 |
break;
|
jacint@1057
|
453 |
case C:
|
jacint@1057
|
454 |
if ( mate[y]!=INVALID ) { //grow
|
jacint@1057
|
455 |
|
jacint@1057
|
456 |
std::cout<<"grow"<<std::endl;
|
jacint@1057
|
457 |
|
jacint@1057
|
458 |
ear.set(y,x);
|
jacint@1057
|
459 |
Node w=mate[y];
|
jacint@1057
|
460 |
blossom.insert(w);
|
jacint@1057
|
461 |
position.set(y,A);
|
jacint@1057
|
462 |
position.set(w,D);
|
jacint@1057
|
463 |
tree.insert(y);
|
jacint@1057
|
464 |
tree.insert(w);
|
jacint@1057
|
465 |
tree.join(y,blossom.find(x));
|
jacint@1057
|
466 |
tree.join(w,y);
|
jacint@1057
|
467 |
Q.push(w);
|
jacint@1057
|
468 |
|
jacint@1057
|
469 |
} else { //augment
|
jacint@1057
|
470 |
|
jacint@1057
|
471 |
std::cout<<"augment"<<std::endl;
|
jacint@1057
|
472 |
|
jacint@1057
|
473 |
augment(x, ear, blossom, tree);
|
jacint@1057
|
474 |
mate.set(x,y);
|
jacint@1057
|
475 |
mate.set(y,x);
|
jacint@1057
|
476 |
return;
|
jacint@1057
|
477 |
} //if
|
jacint@1057
|
478 |
|
jacint@1057
|
479 |
std::cout<<"end c eset"<<std::endl;
|
jacint@1057
|
480 |
break;
|
jacint@1057
|
481 |
default: break;
|
jacint@1057
|
482 |
}
|
jacint@1057
|
483 |
std::cout<<"end switch"<<std::endl;
|
jacint@1057
|
484 |
}
|
jacint@1057
|
485 |
}
|
jacint@1057
|
486 |
}
|
jacint@1057
|
487 |
|
jacint@1057
|
488 |
template <typename Graph>
|
jacint@1057
|
489 |
void MaxMatching<Graph>::greedyMatching() {
|
jacint@1057
|
490 |
for(NodeIt v(g); v!=INVALID; ++v)
|
jacint@1057
|
491 |
if ( mate[v]==INVALID ) {
|
jacint@1057
|
492 |
for( IncEdgeIt e(g,v); e!=INVALID ; ++e ) {
|
jacint@1057
|
493 |
Node y=g.target(e);
|
jacint@1057
|
494 |
if ( mate[y]==INVALID && y!=v ) {
|
jacint@1057
|
495 |
mate.set(v,y);
|
jacint@1057
|
496 |
mate.set(y,v);
|
jacint@1057
|
497 |
break;
|
jacint@1057
|
498 |
}
|
jacint@1057
|
499 |
}
|
jacint@1057
|
500 |
}
|
jacint@1057
|
501 |
}
|
jacint@1057
|
502 |
|
jacint@1057
|
503 |
template <typename Graph>
|
jacint@1057
|
504 |
int MaxMatching<Graph>::size() const {
|
jacint@1057
|
505 |
int s=0;
|
jacint@1057
|
506 |
for(NodeIt v(g); v!=INVALID; ++v) {
|
jacint@1057
|
507 |
if ( mate[v]!=INVALID ) {
|
jacint@1057
|
508 |
++s;
|
jacint@1057
|
509 |
}
|
jacint@1057
|
510 |
}
|
jacint@1057
|
511 |
return (int)s/2;
|
jacint@1057
|
512 |
}
|
jacint@1057
|
513 |
|
jacint@1057
|
514 |
template <typename Graph>
|
jacint@1057
|
515 |
void MaxMatching<Graph>::resetPos() {
|
jacint@1057
|
516 |
for(NodeIt v(g); v!=INVALID; ++v)
|
jacint@1057
|
517 |
position.set(v,C);
|
jacint@1057
|
518 |
}
|
jacint@1057
|
519 |
|
jacint@1057
|
520 |
template <typename Graph>
|
jacint@1057
|
521 |
void MaxMatching<Graph>::resetMatching() {
|
jacint@1057
|
522 |
for(NodeIt v(g); v!=INVALID; ++v)
|
jacint@1057
|
523 |
mate.set(v,INVALID);
|
jacint@1057
|
524 |
}
|
jacint@1057
|
525 |
|
jacint@1057
|
526 |
template <typename Graph>
|
jacint@1057
|
527 |
bool MaxMatching<Graph>::noShrinkStep(Node x, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
528 |
UFE& blossom, UFE& tree, std::queue<Node>& Q) {
|
jacint@1057
|
529 |
for( IncEdgeIt e(g,x); e!= INVALID; ++e ) {
|
jacint@1057
|
530 |
Node y=g.target(e);
|
jacint@1057
|
531 |
|
jacint@1057
|
532 |
if ( position[y]==C ) {
|
jacint@1057
|
533 |
if ( mate[y]!=INVALID ) { //grow
|
jacint@1057
|
534 |
ear.set(y,x);
|
jacint@1057
|
535 |
Node w=mate[y];
|
jacint@1057
|
536 |
blossom.insert(w);
|
jacint@1057
|
537 |
position.set(y,A);
|
jacint@1057
|
538 |
position.set(w,D);
|
jacint@1057
|
539 |
tree.insert(y);
|
jacint@1057
|
540 |
tree.insert(w);
|
jacint@1057
|
541 |
tree.join(y,blossom.find(x));
|
jacint@1057
|
542 |
tree.join(w,y);
|
jacint@1057
|
543 |
Q.push(w);
|
jacint@1057
|
544 |
} else { //augment
|
jacint@1057
|
545 |
augment(x, ear, blossom, tree);
|
jacint@1057
|
546 |
mate.set(x,y);
|
jacint@1057
|
547 |
mate.set(y,x);
|
jacint@1057
|
548 |
return true;
|
jacint@1057
|
549 |
}
|
jacint@1057
|
550 |
}
|
jacint@1057
|
551 |
}
|
jacint@1057
|
552 |
return false;
|
jacint@1057
|
553 |
}
|
jacint@1057
|
554 |
|
jacint@1057
|
555 |
template <typename Graph>
|
jacint@1057
|
556 |
void MaxMatching<Graph>::shrinkStep(Node& top, Node& middle, Node& bottom, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
557 |
UFE& blossom, UFE& tree, std::queue<Node>& Q) {
|
jacint@1057
|
558 |
ear.set(top,bottom);
|
jacint@1057
|
559 |
Node t=top;
|
jacint@1057
|
560 |
while ( t!=middle ) {
|
jacint@1057
|
561 |
Node u=mate[t];
|
jacint@1057
|
562 |
t=ear[u];
|
jacint@1057
|
563 |
ear.set(t,u);
|
jacint@1057
|
564 |
}
|
jacint@1057
|
565 |
bottom=mate[middle];
|
jacint@1057
|
566 |
position.set(bottom,D);
|
jacint@1057
|
567 |
Q.push(bottom);
|
jacint@1057
|
568 |
top=ear[bottom];
|
jacint@1057
|
569 |
Node oldmiddle=middle;
|
jacint@1057
|
570 |
middle=blossom.find(top);
|
jacint@1057
|
571 |
tree.erase(bottom);
|
jacint@1057
|
572 |
tree.erase(oldmiddle);
|
jacint@1057
|
573 |
blossom.insert(bottom);
|
jacint@1057
|
574 |
blossom.join(bottom, oldmiddle);
|
jacint@1057
|
575 |
blossom.join(top, oldmiddle);
|
jacint@1057
|
576 |
}
|
jacint@1057
|
577 |
|
jacint@1057
|
578 |
template <typename Graph>
|
jacint@1057
|
579 |
void MaxMatching<Graph>::augment(Node x, typename Graph::NodeMap<Node>& ear,
|
jacint@1057
|
580 |
UFE& blossom, UFE& tree) {
|
jacint@1057
|
581 |
Node v=mate[x];
|
jacint@1057
|
582 |
while ( v!=INVALID ) {
|
jacint@1057
|
583 |
|
jacint@1057
|
584 |
Node u=ear[v];
|
jacint@1057
|
585 |
mate.set(v,u);
|
jacint@1057
|
586 |
Node tmp=v;
|
jacint@1057
|
587 |
v=mate[u];
|
jacint@1057
|
588 |
mate.set(u,tmp);
|
jacint@1057
|
589 |
}
|
jacint@1057
|
590 |
typename UFE::ItemIt it;
|
jacint@1057
|
591 |
for (tree.first(it,blossom.find(x)); tree.valid(it); tree.next(it)) {
|
jacint@1057
|
592 |
if ( position[it] == D ) {
|
jacint@1057
|
593 |
typename UFE::ItemIt b_it;
|
jacint@1057
|
594 |
for (blossom.first(b_it,it); blossom.valid(b_it); blossom.next(b_it)) {
|
jacint@1057
|
595 |
position.set( b_it ,C);
|
jacint@1057
|
596 |
}
|
jacint@1057
|
597 |
blossom.eraseClass(it);
|
jacint@1057
|
598 |
} else position.set( it ,C);
|
jacint@1057
|
599 |
}
|
jacint@1057
|
600 |
tree.eraseClass(x);
|
jacint@1057
|
601 |
|
jacint@1057
|
602 |
}
|
jacint@1057
|
603 |
|
jacint@1057
|
604 |
/// @}
|
jacint@1057
|
605 |
|
jacint@1057
|
606 |
} //END OF NAMESPACE LEMON
|
jacint@1057
|
607 |
|
jacint@1057
|
608 |
#endif //EDMONDS_H
|