src/work/jacint/matching.h
author athos
Fri, 04 Mar 2005 15:24:07 +0000
changeset 1184 aad134c6c9c5
permissions -rw-r--r--
Corrected an error (dicussed with marci)
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