64 #include <LEP/mcb/config.h> 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> 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> 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> 93 #if defined(LEDA_NAMESPACE) 97 using leda::edge_array;
98 using leda::d_int_set;
102 template<
typename W,
class Container>
106 typedef W result_type;
108 SupportMCB(
const graph& g_,
109 array< Container >& mcb_,
110 array< Container >& proof_,
113 : g(g_), C(mcb_), S(proof_), enumb(enumb_), N(enumb_.dim_cycle_space())
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?)");
123 virtual ~SupportMCB() {}
126 checkPreconditions();
129 float Tcycle = 0.0, Torthog = 0.0, Ttemp;
132 initializeSupportVectors();
135 for(
int k = 0; k < N; ++k ) {
137 leda::used_time( Ttemp );
139 chooseSparsestSupportHeuristic( k );
142 Torthog += leda::used_time( Ttemp );
145 min += computeShortestOddCycle( k );
148 Tcycle += leda::used_time( Ttemp );
150 updateSupportVectors( k );
153 Torthog += leda::used_time( Ttemp );
159 std::cout <<
"LEP_STATS: cycle computation time: " << Tcycle << std::endl;
160 std::cout <<
"LEP_STATS: orthogonal base maintain time: " << Torthog << std::endl;
168 virtual void checkPreconditions() = 0;
170 void initializeSupportVectors() {
171 for(
int i = 0; i < N; ++i ) {
177 void chooseSparsestSupportHeuristic(
int k ) {
178 #ifndef MCB_LEP_UNDIR_NO_EXCHANGE_HEURISTIC 180 for(
int r = k+1; r < N; ++r ) {
181 if ( S[r].size() < S[minS].size() )
185 std::swap( S[k], S[minS] );
190 virtual W computeShortestOddCycle(
int k ) = 0;
192 void updateSupportVectors(
int k ) {
193 for(
int l = k+1; l < N; ++l ) {
194 if ( (C[k].intersect(S[l])).size() % 2 == 1 ) {
208 template<
typename W,
class Container>
209 class WeightedSupportMCB:
public SupportMCB< W, Container>
212 typedef SupportMCB< W, Container> base_type;
214 WeightedSupportMCB(
const graph& g_,
215 const edge_array<W>& len_,
216 array< Container >& mcb_,
217 array< Container >& proof_,
219 : base_type( g_, mcb_, proof_, enumb_ ), len(len_)
225 void checkPreconditions() {
226 #if ! defined(LEDA_CHECKING_OFF) 228 forall_edges( e , g ) {
230 error_handler(999,
"MIN_CYCLE_BASIS: illegal edge (negative weight)");
236 const edge_array<W>& len;
240 template<
typename W,
class Container>
241 class WeightedSignedSupportMCB:
public WeightedSupportMCB< W, Container>
244 typedef WeightedSupportMCB< W, Container> base_type;
246 WeightedSignedSupportMCB(
const graph& g_,
247 const edge_array<W>& len_,
248 array< Container >& mcb_,
249 array< Container >& proof_,
251 : base_type( g_, len_, mcb_, proof_, enumb_ ), sg(g_, len_, enumb_)
257 W computeShortestOddCycle(
int k ) {
258 return sg.get_shortest_odd_cycle( S[k], C[k] );
262 detail::WeightedSignedGraph<W,leda::bin_heap> sg;
267 template<
class Container>
268 class UnweightedSignedSupportMCB:
public SupportMCB< int, Container >
271 typedef SupportMCB< int, Container> base_type;
273 UnweightedSignedSupportMCB(
const graph& g_,
274 array< Container >& mcb_,
275 array< Container >& proof_,
278 : base_type( g_, mcb_, proof_, enumb_ ), sg( g_, enumb_ )
282 ~UnweightedSignedSupportMCB() {}
285 int computeShortestOddCycle(
int k ) {
286 return sg.get_shortest_odd_cycle( S[k], C[k] );
289 void checkPreconditions() {}
292 detail::UnweightedSignedGraph sg;
297 template<
typename W,
class Container>
298 class WeightedHortonSupportMCB:
public WeightedSupportMCB<W, Container>
301 typedef WeightedSupportMCB<W, Container> base_type;
303 WeightedHortonSupportMCB(
const graph& g_,
304 const edge_array<W>& len_,
305 array< Container >& mcb_,
306 array< Container >& proof_,
308 : base_type( g_, len_, mcb_, proof_, enumb_ ), HS( g_, len_, enumb_ )
314 W computeShortestOddCycle(
int k ) {
315 return HS.get_shortest_odd_cycle( S[k], C[k] );
318 HortonSuperset<W> HS;
323 template<
typename W,
class Container>
324 class WeightedSPTreesSupportMCB:
public WeightedSupportMCB<W, Container>
327 typedef WeightedSupportMCB<W, Container> base_type;
329 WeightedSPTreesSupportMCB(
const graph& g_,
330 const edge_array<W>& len_,
331 array< Container >& mcb_,
332 array< Container >& proof_,
334 : base_type( g_, len_, mcb_, proof_, enumb_ ), uhst( g_, len_, enumb_ )
340 W computeShortestOddCycle(
int k ) {
341 uhst.updateTreeLabels( S[k] );
342 return uhst.getShortestOddCycle( S[k], C[k] );
345 UndirectedHortonSupersetTrees<W> uhst;
397 template<
class Container>
399 array< Container >&
mcb,
400 array< Container >& proof,
404 UnweightedSignedSupportMCB<Container> tmp( g, mcb, proof, enumb );
432 template<
class Container>
434 array< Container >&
mcb,
438 array< Container > proof;
439 return UMCB_SVA( g, mcb, proof, enumb );
474 template<
class W,
class Container>
476 const edge_array<W>& len,
477 array< Container >&
mcb,
478 array< Container >& proof,
482 WeightedSignedSupportMCB<W,Container> tmp( g, len, mcb, proof, enumb );
514 template<
class W,
class Container>
516 const edge_array<W>& len,
517 array< Container >&
mcb,
521 array< Container > proof;
522 WeightedSignedSupportMCB<W,Container> tmp( g, len, mcb, proof, enumb );
554 leda::array< leda::d_int_set >&
mcb,
555 leda::array< leda::d_int_set >& proof,
581 leda::array< leda::d_int_set >& mcb,
617 const edge_array<W>& len,
618 array< d_int_set >& mcb,
619 array< d_int_set >& proof,
623 WeightedHortonSupportMCB<W, d_int_set> tmp( g, len, mcb, proof, enumb );
653 const edge_array<W>& len,
654 array< d_int_set >& mcb,
658 array< d_int_set > proof_temp;
659 return UMCB_HYBRID( g, len, mcb, proof_temp, enumb );
694 const edge_array<W>& len,
695 array< mcb::spvecgf2 >& mcb,
696 array< mcb::spvecgf2 >& proof,
699 WeightedSPTreesSupportMCB<W,mcb::spvecgf2> tmp( g, len, mcb, proof, enumb );
731 const edge_array<W>& len,
732 array< mcb::spvecgf2 >& mcb,
735 array< mcb::spvecgf2 > proof;
736 WeightedSPTreesSupportMCB<W,mcb::spvecgf2> tmp( g, len, mcb, proof, enumb );
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'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