mcb LEDA Extension Package  0.8
umcb.h
Go to the documentation of this file.
1 //
2 // This program can be freely used in an academic environment
3 // ONLY for research purposes, subject to the following restrictions:
4 //
5 // 1. The origin of this software must not be misrepresented; you must not
6 // claim that you wrote the original software. If you use this software
7 // an acknowledgment in the product documentation is required.
8 // 2. Altered source versions must be plainly marked as such, and must not be
9 // misrepresented as being the original software.
10 // 3. This notice may not be removed or altered from any source distribution.
11 //
12 // Any other use is strictly prohibited by the author, without an explicit
13 // permission.
14 //
15 // Note that this program uses the LEDA library, which is NOT free. For more
16 // details visit Algorithmic Solutions at http://www.algorithmic-solutions.com/
17 // There is also a free version of LEDA 6.0 or newer.
18 //
19 // !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
20 // ! Any commercial use of this software is strictly !
21 // ! prohibited without explicit permission by the !
22 // ! author. !
23 // !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
24 //
25 // This software is distributed in the hope that it will be useful,
26 // but WITHOUT ANY WARRANTY; without even the implied warranty of
27 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
28 //
29 // Copyright (C) 2004-2008 - Dimitrios Michail <dimitrios.michail@gmail.com>
30 //
31 
61 #ifndef UMCB_H
62 #define UMCB_H
63 
64 #include <LEP/mcb/config.h>
65 
66 #ifdef LEDA_GE_V5
67 #include <LEDA/graph/graph.h>
68 #include <LEDA/graph/edge_array.h>
69 #include <LEDA/graph/node_array.h>
70 #include <LEDA/core/d_int_set.h>
71 #include <LEDA/core/array.h>
72 #include <LEDA/core/list.h>
73 #else
74 #include <LEDA/graph.h>
75 #include <LEDA/d_int_set.h>
76 #include <LEDA/edge_array.h>
77 #include <LEDA/node_array.h>
78 #include <LEDA/array.h>
79 #include <LEDA/list.h>
80 #endif
81 
82 #include <LEP/mcb/spvecgf2.h>
83 #include <LEP/mcb/signed.h>
84 #include <LEP/mcb/superset.h>
85 #include <LEP/mcb/sptrees.h>
86 #include <LEP/mcb/transform.h>
87 #include <LEP/mcb/verify.h>
88 
89 // start our namespace
90 namespace mcb
91 {
92 
93 #if defined(LEDA_NAMESPACE)
94  using leda::graph;
95  using leda::array;
96  using leda::edge;
97  using leda::edge_array;
98  using leda::d_int_set;
99  using leda::list;
100 #endif
101 
102  template<typename W, class Container>
103  class SupportMCB
104  {
105  public:
106  typedef W result_type;
107 
108  SupportMCB( const graph& g_,
109  array< Container >& mcb_,
110  array< Container >& proof_,
111  const mcb::edge_num& enumb_
112  )
113  : g(g_), C(mcb_), S(proof_), enumb(enumb_), N(enumb_.dim_cycle_space())
114  {
115 #if ! defined(LEDA_CHECKING_OFF)
116  if ( Is_Undirected_Simple( g ) == false )
117  error_handler(999,"MIN_CYCLE_BASIS: illegal graph (parallel,anti-parallel edges or loops?)");
118 #endif
119  C.resize(N);
120  S.resize(N);
121  }
122 
123  virtual ~SupportMCB() {}
124 
125  W run() {
126  checkPreconditions();
127 
128 #ifdef LEP_STATS
129  float Tcycle = 0.0, Torthog = 0.0, Ttemp;
130 #endif
131 
132  initializeSupportVectors();
133 
134  W min = W();
135  for( int k = 0; k < N; ++k ) {
136 #ifdef LEP_STATS
137  leda::used_time( Ttemp );
138 #endif
139  chooseSparsestSupportHeuristic( k );
140 
141 #ifdef LEP_STATS
142  Torthog += leda::used_time( Ttemp );
143 #endif
144 
145  min += computeShortestOddCycle( k );
146 
147 #ifdef LEP_STATS
148  Tcycle += leda::used_time( Ttemp );
149 #endif
150  updateSupportVectors( k );
151 
152 #ifdef LEP_STATS
153  Torthog += leda::used_time( Ttemp );
154 #endif
155 
156  }
157 
158 #ifdef LEP_STATS
159  std::cout << "LEP_STATS: cycle computation time: " << Tcycle << std::endl;
160  std::cout << "LEP_STATS: orthogonal base maintain time: " << Torthog << std::endl;
161 #endif
162 
163  return min;
164  }
165 
166  private:
167 
168  virtual void checkPreconditions() = 0;
169 
170  void initializeSupportVectors() {
171  for( int i = 0; i < N; ++i ) {
172  S[i].clear();
173  S[i].insert(i);
174  }
175  }
176 
177  void chooseSparsestSupportHeuristic( int k ) {
178 #ifndef MCB_LEP_UNDIR_NO_EXCHANGE_HEURISTIC
179  int minS = k;
180  for( int r = k+1; r < N; ++r ) {
181  if ( S[r].size() < S[minS].size() )
182  minS = r;
183  }
184  if ( minS != k ) { // swap
185  std::swap( S[k], S[minS] );
186  }
187 #endif
188  }
189 
190  virtual W computeShortestOddCycle( int k ) = 0;
191 
192  void updateSupportVectors( int k ) {
193  for( int l = k+1; l < N; ++l ) {
194  if ( (C[k].intersect(S[l])).size() % 2 == 1 ) {
195  S[ l ] %= S[k];
196  }
197  }
198  }
199 
200  protected:
201  const graph& g;
202  array<Container>& C;
203  array<Container>& S;
204  const mcb::edge_num& enumb;
205  int N;
206  };
207 
208  template<typename W, class Container>
209  class WeightedSupportMCB: public SupportMCB< W, Container>
210  {
211  public:
212  typedef SupportMCB< W, Container> base_type;
213 
214  WeightedSupportMCB( const graph& g_,
215  const edge_array<W>& len_,
216  array< Container >& mcb_,
217  array< Container >& proof_,
218  const mcb::edge_num& enumb_ )
219  : base_type( g_, mcb_, proof_, enumb_ ), len(len_)
220  {
221  }
222 
223  private:
224 
225  void checkPreconditions() {
226 #if ! defined(LEDA_CHECKING_OFF)
227  edge e;
228  forall_edges( e , g ) {
229  if ( len[e] < 0 )
230  error_handler(999,"MIN_CYCLE_BASIS: illegal edge (negative weight)");
231  }
232 #endif
233  }
234 
235  protected:
236  const edge_array<W>& len;
237  using base_type::g;
238  };
239 
240  template<typename W, class Container>
241  class WeightedSignedSupportMCB: public WeightedSupportMCB< W, Container>
242  {
243  public:
244  typedef WeightedSupportMCB< W, Container> base_type;
245 
246  WeightedSignedSupportMCB( const graph& g_,
247  const edge_array<W>& len_,
248  array< Container >& mcb_,
249  array< Container >& proof_,
250  const mcb::edge_num& enumb_ )
251  : base_type( g_, len_, mcb_, proof_, enumb_ ), sg(g_, len_, enumb_)
252  {
253  }
254 
255  private:
256 
257  W computeShortestOddCycle( int k ) {
258  return sg.get_shortest_odd_cycle( S[k], C[k] );
259  }
260 
261  private:
262  detail::WeightedSignedGraph<W,leda::bin_heap> sg;
263  using base_type::C;
264  using base_type::S;
265  };
266 
267  template<class Container>
268  class UnweightedSignedSupportMCB: public SupportMCB< int, Container >
269  {
270  public:
271  typedef SupportMCB< int, Container> base_type;
272 
273  UnweightedSignedSupportMCB( const graph& g_,
274  array< Container >& mcb_,
275  array< Container >& proof_,
276  const mcb::edge_num& enumb_
277  )
278  : base_type( g_, mcb_, proof_, enumb_ ), sg( g_, enumb_ )
279  {
280  }
281 
282  ~UnweightedSignedSupportMCB() {}
283 
284  private:
285  int computeShortestOddCycle( int k ) {
286  return sg.get_shortest_odd_cycle( S[k], C[k] );
287  }
288 
289  void checkPreconditions() {}
290 
291  private:
292  detail::UnweightedSignedGraph sg;
293  using base_type::C;
294  using base_type::S;
295  };
296 
297  template<typename W, class Container>
298  class WeightedHortonSupportMCB: public WeightedSupportMCB<W, Container>
299  {
300  public:
301  typedef WeightedSupportMCB<W, Container> base_type;
302 
303  WeightedHortonSupportMCB( const graph& g_,
304  const edge_array<W>& len_,
305  array< Container >& mcb_,
306  array< Container >& proof_,
307  const mcb::edge_num& enumb_ )
308  : base_type( g_, len_, mcb_, proof_, enumb_ ), HS( g_, len_, enumb_ )
309  {
310  }
311 
312  private:
313 
314  W computeShortestOddCycle( int k ) {
315  return HS.get_shortest_odd_cycle( S[k], C[k] );
316  }
317 
318  HortonSuperset<W> HS;
319  using base_type::C;
320  using base_type::S;
321  };
322 
323  template<typename W, class Container>
324  class WeightedSPTreesSupportMCB: public WeightedSupportMCB<W, Container>
325  {
326  public:
327  typedef WeightedSupportMCB<W, Container> base_type;
328 
329  WeightedSPTreesSupportMCB( const graph& g_,
330  const edge_array<W>& len_,
331  array< Container >& mcb_,
332  array< Container >& proof_,
333  const mcb::edge_num& enumb_ )
334  : base_type( g_, len_, mcb_, proof_, enumb_ ), uhst( g_, len_, enumb_ )
335  {
336  }
337 
338  private:
339 
340  W computeShortestOddCycle( int k ) {
341  uhst.updateTreeLabels( S[k] );
342  return uhst.getShortestOddCycle( S[k], C[k] );
343  }
344 
345  UndirectedHortonSupersetTrees<W> uhst;
346  using base_type::C;
347  using base_type::S;
348  };
349 
350 
351 
368 
397  template<class Container>
398  int UMCB_SVA( const graph& g,
399  array< Container >& mcb,
400  array< Container >& proof,
401  const mcb::edge_num& enumb
402  )
403  {
404  UnweightedSignedSupportMCB<Container> tmp( g, mcb, proof, enumb );
405  return tmp.run();
406  }
407 
432  template<class Container>
433  int UMCB_SVA( const graph& g,
434  array< Container >& mcb,
435  const mcb::edge_num& enumb
436  )
437  {
438  array< Container > proof;
439  return UMCB_SVA( g, mcb, proof, enumb );
440  }
441 
474  template<class W, class Container>
475  W UMCB_SVA( const graph& g,
476  const edge_array<W>& len,
477  array< Container >& mcb,
478  array< Container >& proof,
479  const mcb::edge_num& enumb
480  )
481  {
482  WeightedSignedSupportMCB<W,Container> tmp( g, len, mcb, proof, enumb );
483  return tmp.run();
484  }
485 
514  template<class W, class Container>
515  W UMCB_SVA( const graph& g,
516  const edge_array<W>& len,
517  array< Container >& mcb,
518  const mcb::edge_num& enumb
519  )
520  {
521  array< Container > proof;
522  WeightedSignedSupportMCB<W,Container> tmp( g, len, mcb, proof, enumb );
523  return tmp.run();
524  }
525 
553  extern int UMCB_HYBRID( const leda::graph& g,
554  leda::array< leda::d_int_set >& mcb,
555  leda::array< leda::d_int_set >& proof,
556  const mcb::edge_num& enumb
557  );
558 
580  extern int UMCB_HYBRID( const leda::graph& g,
581  leda::array< leda::d_int_set >& mcb,
582  const mcb::edge_num& enumb
583  );
584 
585 
615  template<class W>
616  W UMCB_HYBRID( const graph& g,
617  const edge_array<W>& len,
618  array< d_int_set >& mcb,
619  array< d_int_set >& proof,
620  const mcb::edge_num& enumb
621  )
622  {
623  WeightedHortonSupportMCB<W, d_int_set> tmp( g, len, mcb, proof, enumb );
624  return tmp.run();
625  }
626 
627 
651  template<class W>
652  W UMCB_HYBRID( const graph& g,
653  const edge_array<W>& len,
654  array< d_int_set >& mcb,
655  const mcb::edge_num& enumb
656  )
657  {
658  array< d_int_set > proof_temp;
659  return UMCB_HYBRID( g, len, mcb, proof_temp, enumb );
660  }
661 
692  template<class W>
693  W UMCB_FH( const graph& g,
694  const edge_array<W>& len,
695  array< mcb::spvecgf2 >& mcb,
696  array< mcb::spvecgf2 >& proof,
697  const mcb::edge_num& enumb )
698  {
699  WeightedSPTreesSupportMCB<W,mcb::spvecgf2> tmp( g, len, mcb, proof, enumb );
700  return tmp.run();
701  }
702 
729  template<class W>
730  W UMCB_FH( const graph& g,
731  const edge_array<W>& len,
732  array< mcb::spvecgf2 >& mcb,
733  const mcb::edge_num& enumb )
734  {
735  array< mcb::spvecgf2 > proof;
736  WeightedSPTreesSupportMCB<W,mcb::spvecgf2> tmp( g, len, mcb, proof, enumb );
737  return tmp.run();
738  }
739 
741 
742 } // namespace mcb end
743 
744 #endif // UMCB_H
745 
746 /* ex: set ts=4 sw=4 sts=4 et: */
747 
748 
Definitions of sparse vector.
An edge numbering class.
Definition: edge_num.h:75
int UMCB_SVA(const graph &g, array< Container > &mcb, array< Container > &proof, const mcb::edge_num &enumb)
Compute a MCB of an undirected graph using the Support Vector Approach (de Pina&#39;s PhD thesis) algorit...
Definition: umcb.h:398
The main package namespace.
int UMCB_HYBRID(const leda::graph &g, leda::array< leda::d_int_set > &mcb, leda::array< leda::d_int_set > &proof, const mcb::edge_num &enumb)
Compute a minimum cycle basis of an undirected graph using a hybrid algorithm.
W UMCB_FH(const graph &g, const edge_array< W > &len, array< mcb::spvecgf2 > &mcb, array< mcb::spvecgf2 > &proof, const mcb::edge_num &enumb)
Compute a MCB of an undirected weighted graph using a Fast variant of the Hybrid algorithm.
Definition: umcb.h:693