lemon/cycle_canceling.h
author alpar
Thu, 23 Oct 2008 16:49:06 +0000
changeset 2626 324cfbf66a12
parent 2620 8f41a3129746
child 2629 84354c78b068
permissions -rw-r--r--
Adapt the doc generation to the hg repo
deba@2440
     1
/* -*- C++ -*-
deba@2440
     2
 *
deba@2440
     3
 * This file is a part of LEMON, a generic C++ optimization library
deba@2440
     4
 *
alpar@2553
     5
 * Copyright (C) 2003-2008
deba@2440
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
deba@2440
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
deba@2440
     8
 *
deba@2440
     9
 * Permission to use, modify and distribute this software is granted
deba@2440
    10
 * provided that this copyright notice appears in all copies. For
deba@2440
    11
 * precise terms see the accompanying LICENSE file.
deba@2440
    12
 *
deba@2440
    13
 * This software is provided "AS IS" with no warranty of any kind,
deba@2440
    14
 * express or implied, and with no claim as to its suitability for any
deba@2440
    15
 * purpose.
deba@2440
    16
 *
deba@2440
    17
 */
deba@2440
    18
deba@2440
    19
#ifndef LEMON_CYCLE_CANCELING_H
deba@2440
    20
#define LEMON_CYCLE_CANCELING_H
deba@2440
    21
deba@2440
    22
/// \ingroup min_cost_flow
deba@2440
    23
///
deba@2440
    24
/// \file
kpeter@2573
    25
/// \brief Cycle-canceling algorithm for finding a minimum cost flow.
deba@2440
    26
deba@2440
    27
#include <vector>
kpeter@2509
    28
#include <lemon/graph_adaptor.h>
kpeter@2573
    29
#include <lemon/path.h>
kpeter@2573
    30
deba@2440
    31
#include <lemon/circulation.h>
kpeter@2573
    32
#include <lemon/bellman_ford.h>
kpeter@2573
    33
#include <lemon/min_mean_cycle.h>
kpeter@2556
    34
deba@2440
    35
namespace lemon {
deba@2440
    36
deba@2440
    37
  /// \addtogroup min_cost_flow
deba@2440
    38
  /// @{
deba@2440
    39
kpeter@2556
    40
  /// \brief Implementation of a cycle-canceling algorithm for
kpeter@2556
    41
  /// finding a minimum cost flow.
deba@2440
    42
  ///
kpeter@2556
    43
  /// \ref CycleCanceling implements a cycle-canceling algorithm for
kpeter@2556
    44
  /// finding a minimum cost flow.
deba@2440
    45
  ///
kpeter@2573
    46
  /// \tparam Graph The directed graph type the algorithm runs on.
kpeter@2573
    47
  /// \tparam LowerMap The type of the lower bound map.
kpeter@2573
    48
  /// \tparam CapacityMap The type of the capacity (upper bound) map.
kpeter@2573
    49
  /// \tparam CostMap The type of the cost (length) map.
kpeter@2573
    50
  /// \tparam SupplyMap The type of the supply map.
deba@2440
    51
  ///
deba@2440
    52
  /// \warning
kpeter@2573
    53
  /// - Edge capacities and costs should be \e non-negative \e integers.
kpeter@2573
    54
  /// - Supply values should be \e signed \e integers.
kpeter@2581
    55
  /// - The value types of the maps should be convertible to each other.
kpeter@2581
    56
  /// - \c CostMap::Value must be signed type.
kpeter@2573
    57
  ///
kpeter@2573
    58
  /// \note By default the \ref BellmanFord "Bellman-Ford" algorithm is
kpeter@2573
    59
  /// used for negative cycle detection with limited iteration number.
kpeter@2573
    60
  /// However \ref CycleCanceling also provides the "Minimum Mean
kpeter@2573
    61
  /// Cycle-Canceling" algorithm, which is \e strongly \e polynomial,
kpeter@2573
    62
  /// but rather slower in practice.
kpeter@2573
    63
  /// To use this version of the algorithm, call \ref run() with \c true
kpeter@2573
    64
  /// parameter.
deba@2440
    65
  ///
deba@2440
    66
  /// \author Peter Kovacs
kpeter@2533
    67
  template < typename Graph,
kpeter@2533
    68
             typename LowerMap = typename Graph::template EdgeMap<int>,
kpeter@2573
    69
             typename CapacityMap = typename Graph::template EdgeMap<int>,
kpeter@2533
    70
             typename CostMap = typename Graph::template EdgeMap<int>,
kpeter@2573
    71
             typename SupplyMap = typename Graph::template NodeMap<int> >
deba@2440
    72
  class CycleCanceling
deba@2440
    73
  {
kpeter@2556
    74
    GRAPH_TYPEDEFS(typename Graph);
deba@2440
    75
deba@2440
    76
    typedef typename CapacityMap::Value Capacity;
deba@2440
    77
    typedef typename CostMap::Value Cost;
deba@2440
    78
    typedef typename SupplyMap::Value Supply;
kpeter@2556
    79
    typedef typename Graph::template EdgeMap<Capacity> CapacityEdgeMap;
kpeter@2556
    80
    typedef typename Graph::template NodeMap<Supply> SupplyNodeMap;
deba@2440
    81
deba@2440
    82
    typedef ResGraphAdaptor< const Graph, Capacity,
kpeter@2556
    83
                             CapacityEdgeMap, CapacityEdgeMap > ResGraph;
deba@2440
    84
    typedef typename ResGraph::Node ResNode;
deba@2440
    85
    typedef typename ResGraph::NodeIt ResNodeIt;
deba@2440
    86
    typedef typename ResGraph::Edge ResEdge;
deba@2440
    87
    typedef typename ResGraph::EdgeIt ResEdgeIt;
deba@2440
    88
deba@2440
    89
  public:
deba@2440
    90
kpeter@2556
    91
    /// The type of the flow map.
kpeter@2556
    92
    typedef typename Graph::template EdgeMap<Capacity> FlowMap;
kpeter@2581
    93
    /// The type of the potential map.
kpeter@2581
    94
    typedef typename Graph::template NodeMap<Cost> PotentialMap;
deba@2440
    95
kpeter@2573
    96
  private:
deba@2440
    97
kpeter@2573
    98
    /// \brief Map adaptor class for handling residual edge costs.
kpeter@2573
    99
    ///
kpeter@2620
   100
    /// Map adaptor class for handling residual edge costs.
kpeter@2573
   101
    class ResidualCostMap : public MapBase<ResEdge, Cost>
deba@2440
   102
    {
deba@2440
   103
    private:
deba@2440
   104
kpeter@2573
   105
      const CostMap &_cost_map;
deba@2440
   106
deba@2440
   107
    public:
deba@2440
   108
kpeter@2573
   109
      ///\e
kpeter@2573
   110
      ResidualCostMap(const CostMap &cost_map) : _cost_map(cost_map) {}
deba@2440
   111
kpeter@2573
   112
      ///\e
kpeter@2509
   113
      Cost operator[](const ResEdge &e) const {
kpeter@2573
   114
        return ResGraph::forward(e) ? _cost_map[e] : -_cost_map[e];
deba@2440
   115
      }
deba@2440
   116
kpeter@2573
   117
    }; //class ResidualCostMap
deba@2440
   118
kpeter@2573
   119
  private:
deba@2440
   120
kpeter@2573
   121
    // The maximum number of iterations for the first execution of the
kpeter@2573
   122
    // Bellman-Ford algorithm. It should be at least 2.
kpeter@2593
   123
    static const int BF_FIRST_LIMIT  = 2;
kpeter@2573
   124
    // The iteration limit for the Bellman-Ford algorithm is multiplied
kpeter@2593
   125
    // by BF_LIMIT_FACTOR/100 in every round.
kpeter@2593
   126
    static const int BF_LIMIT_FACTOR = 150;
deba@2440
   127
kpeter@2573
   128
  private:
deba@2440
   129
kpeter@2573
   130
    // The directed graph the algorithm runs on
kpeter@2573
   131
    const Graph &_graph;
kpeter@2573
   132
    // The original lower bound map
kpeter@2573
   133
    const LowerMap *_lower;
kpeter@2573
   134
    // The modified capacity map
kpeter@2573
   135
    CapacityEdgeMap _capacity;
kpeter@2573
   136
    // The original cost map
kpeter@2573
   137
    const CostMap &_cost;
kpeter@2573
   138
    // The modified supply map
kpeter@2573
   139
    SupplyNodeMap _supply;
kpeter@2573
   140
    bool _valid_supply;
kpeter@2573
   141
kpeter@2573
   142
    // Edge map of the current flow
kpeter@2581
   143
    FlowMap *_flow;
kpeter@2581
   144
    bool _local_flow;
kpeter@2581
   145
    // Node map of the current potentials
kpeter@2581
   146
    PotentialMap *_potential;
kpeter@2581
   147
    bool _local_potential;
kpeter@2573
   148
kpeter@2573
   149
    // The residual graph
kpeter@2581
   150
    ResGraph *_res_graph;
kpeter@2573
   151
    // The residual cost map
kpeter@2573
   152
    ResidualCostMap _res_cost;
kpeter@2573
   153
kpeter@2573
   154
  public:
deba@2440
   155
kpeter@2581
   156
    /// \brief General constructor (with lower bounds).
deba@2440
   157
    ///
kpeter@2581
   158
    /// General constructor (with lower bounds).
deba@2440
   159
    ///
kpeter@2573
   160
    /// \param graph The directed graph the algorithm runs on.
kpeter@2573
   161
    /// \param lower The lower bounds of the edges.
kpeter@2573
   162
    /// \param capacity The capacities (upper bounds) of the edges.
kpeter@2573
   163
    /// \param cost The cost (length) values of the edges.
kpeter@2573
   164
    /// \param supply The supply values of the nodes (signed).
kpeter@2573
   165
    CycleCanceling( const Graph &graph,
kpeter@2573
   166
                    const LowerMap &lower,
kpeter@2573
   167
                    const CapacityMap &capacity,
kpeter@2573
   168
                    const CostMap &cost,
kpeter@2573
   169
                    const SupplyMap &supply ) :
kpeter@2573
   170
      _graph(graph), _lower(&lower), _capacity(graph), _cost(cost),
kpeter@2623
   171
      _supply(graph), _flow(NULL), _local_flow(false),
kpeter@2623
   172
      _potential(NULL), _local_potential(false), _res_graph(NULL),
kpeter@2623
   173
      _res_cost(_cost)
deba@2440
   174
    {
kpeter@2556
   175
      // Removing non-zero lower bounds
kpeter@2573
   176
      _capacity = subMap(capacity, lower);
deba@2440
   177
      Supply sum = 0;
kpeter@2573
   178
      for (NodeIt n(_graph); n != INVALID; ++n) {
kpeter@2573
   179
        Supply s = supply[n];
kpeter@2573
   180
        for (InEdgeIt e(_graph, n); e != INVALID; ++e)
kpeter@2573
   181
          s += lower[e];
kpeter@2573
   182
        for (OutEdgeIt e(_graph, n); e != INVALID; ++e)
kpeter@2573
   183
          s -= lower[e];
kpeter@2573
   184
        sum += (_supply[n] = s);
deba@2440
   185
      }
kpeter@2573
   186
      _valid_supply = sum == 0;
deba@2440
   187
    }
deba@2440
   188
kpeter@2581
   189
    /// \brief General constructor (without lower bounds).
deba@2440
   190
    ///
kpeter@2581
   191
    /// General constructor (without lower bounds).
deba@2440
   192
    ///
kpeter@2573
   193
    /// \param graph The directed graph the algorithm runs on.
kpeter@2573
   194
    /// \param capacity The capacities (upper bounds) of the edges.
kpeter@2573
   195
    /// \param cost The cost (length) values of the edges.
kpeter@2573
   196
    /// \param supply The supply values of the nodes (signed).
kpeter@2573
   197
    CycleCanceling( const Graph &graph,
kpeter@2573
   198
                    const CapacityMap &capacity,
kpeter@2573
   199
                    const CostMap &cost,
kpeter@2573
   200
                    const SupplyMap &supply ) :
kpeter@2573
   201
      _graph(graph), _lower(NULL), _capacity(capacity), _cost(cost),
kpeter@2623
   202
      _supply(supply), _flow(NULL), _local_flow(false),
kpeter@2623
   203
      _potential(NULL), _local_potential(false), _res_graph(NULL),
kpeter@2623
   204
      _res_cost(_cost)
deba@2440
   205
    {
deba@2440
   206
      // Checking the sum of supply values
deba@2440
   207
      Supply sum = 0;
kpeter@2573
   208
      for (NodeIt n(_graph); n != INVALID; ++n) sum += _supply[n];
kpeter@2573
   209
      _valid_supply = sum == 0;
deba@2440
   210
    }
deba@2440
   211
kpeter@2581
   212
    /// \brief Simple constructor (with lower bounds).
deba@2440
   213
    ///
kpeter@2581
   214
    /// Simple constructor (with lower bounds).
deba@2440
   215
    ///
kpeter@2573
   216
    /// \param graph The directed graph the algorithm runs on.
kpeter@2573
   217
    /// \param lower The lower bounds of the edges.
kpeter@2573
   218
    /// \param capacity The capacities (upper bounds) of the edges.
kpeter@2573
   219
    /// \param cost The cost (length) values of the edges.
kpeter@2573
   220
    /// \param s The source node.
kpeter@2573
   221
    /// \param t The target node.
kpeter@2573
   222
    /// \param flow_value The required amount of flow from node \c s
kpeter@2573
   223
    /// to node \c t (i.e. the supply of \c s and the demand of \c t).
kpeter@2573
   224
    CycleCanceling( const Graph &graph,
kpeter@2573
   225
                    const LowerMap &lower,
kpeter@2573
   226
                    const CapacityMap &capacity,
kpeter@2573
   227
                    const CostMap &cost,
kpeter@2573
   228
                    Node s, Node t,
kpeter@2573
   229
                    Supply flow_value ) :
kpeter@2573
   230
      _graph(graph), _lower(&lower), _capacity(graph), _cost(cost),
kpeter@2623
   231
      _supply(graph), _flow(NULL), _local_flow(false),
kpeter@2623
   232
      _potential(NULL), _local_potential(false), _res_graph(NULL),
kpeter@2623
   233
      _res_cost(_cost)
deba@2440
   234
    {
kpeter@2556
   235
      // Removing non-zero lower bounds
kpeter@2573
   236
      _capacity = subMap(capacity, lower);
kpeter@2573
   237
      for (NodeIt n(_graph); n != INVALID; ++n) {
kpeter@2573
   238
        Supply sum = 0;
kpeter@2573
   239
        if (n == s) sum =  flow_value;
kpeter@2573
   240
        if (n == t) sum = -flow_value;
kpeter@2573
   241
        for (InEdgeIt e(_graph, n); e != INVALID; ++e)
kpeter@2573
   242
          sum += lower[e];
kpeter@2573
   243
        for (OutEdgeIt e(_graph, n); e != INVALID; ++e)
kpeter@2573
   244
          sum -= lower[e];
kpeter@2573
   245
        _supply[n] = sum;
deba@2440
   246
      }
kpeter@2573
   247
      _valid_supply = true;
deba@2440
   248
    }
deba@2440
   249
kpeter@2581
   250
    /// \brief Simple constructor (without lower bounds).
deba@2440
   251
    ///
kpeter@2581
   252
    /// Simple constructor (without lower bounds).
deba@2440
   253
    ///
kpeter@2573
   254
    /// \param graph The directed graph the algorithm runs on.
kpeter@2573
   255
    /// \param capacity The capacities (upper bounds) of the edges.
kpeter@2573
   256
    /// \param cost The cost (length) values of the edges.
kpeter@2573
   257
    /// \param s The source node.
kpeter@2573
   258
    /// \param t The target node.
kpeter@2573
   259
    /// \param flow_value The required amount of flow from node \c s
kpeter@2573
   260
    /// to node \c t (i.e. the supply of \c s and the demand of \c t).
kpeter@2573
   261
    CycleCanceling( const Graph &graph,
kpeter@2573
   262
                    const CapacityMap &capacity,
kpeter@2573
   263
                    const CostMap &cost,
kpeter@2573
   264
                    Node s, Node t,
kpeter@2573
   265
                    Supply flow_value ) :
kpeter@2573
   266
      _graph(graph), _lower(NULL), _capacity(capacity), _cost(cost),
kpeter@2623
   267
      _supply(graph, 0), _flow(NULL), _local_flow(false),
kpeter@2623
   268
      _potential(NULL), _local_potential(false), _res_graph(NULL),
kpeter@2623
   269
      _res_cost(_cost)
deba@2440
   270
    {
kpeter@2573
   271
      _supply[s] =  flow_value;
kpeter@2573
   272
      _supply[t] = -flow_value;
kpeter@2573
   273
      _valid_supply = true;
deba@2440
   274
    }
deba@2440
   275
kpeter@2581
   276
    /// Destructor.
kpeter@2581
   277
    ~CycleCanceling() {
kpeter@2581
   278
      if (_local_flow) delete _flow;
kpeter@2581
   279
      if (_local_potential) delete _potential;
kpeter@2581
   280
      delete _res_graph;
kpeter@2581
   281
    }
kpeter@2581
   282
kpeter@2620
   283
    /// \brief Set the flow map.
kpeter@2581
   284
    ///
kpeter@2620
   285
    /// Set the flow map.
kpeter@2581
   286
    ///
kpeter@2581
   287
    /// \return \c (*this)
kpeter@2581
   288
    CycleCanceling& flowMap(FlowMap &map) {
kpeter@2581
   289
      if (_local_flow) {
kpeter@2581
   290
        delete _flow;
kpeter@2581
   291
        _local_flow = false;
kpeter@2581
   292
      }
kpeter@2581
   293
      _flow = &map;
kpeter@2581
   294
      return *this;
kpeter@2581
   295
    }
kpeter@2581
   296
kpeter@2620
   297
    /// \brief Set the potential map.
kpeter@2581
   298
    ///
kpeter@2620
   299
    /// Set the potential map.
kpeter@2581
   300
    ///
kpeter@2581
   301
    /// \return \c (*this)
kpeter@2581
   302
    CycleCanceling& potentialMap(PotentialMap &map) {
kpeter@2581
   303
      if (_local_potential) {
kpeter@2581
   304
        delete _potential;
kpeter@2581
   305
        _local_potential = false;
kpeter@2581
   306
      }
kpeter@2581
   307
      _potential = &map;
kpeter@2581
   308
      return *this;
kpeter@2581
   309
    }
kpeter@2581
   310
kpeter@2581
   311
    /// \name Execution control
kpeter@2581
   312
kpeter@2581
   313
    /// @{
kpeter@2581
   314
kpeter@2620
   315
    /// \brief Run the algorithm.
kpeter@2556
   316
    ///
kpeter@2620
   317
    /// Run the algorithm.
kpeter@2556
   318
    ///
kpeter@2573
   319
    /// \param min_mean_cc Set this parameter to \c true to run the
kpeter@2573
   320
    /// "Minimum Mean Cycle-Canceling" algorithm, which is strongly
kpeter@2573
   321
    /// polynomial, but rather slower in practice.
kpeter@2573
   322
    ///
kpeter@2556
   323
    /// \return \c true if a feasible flow can be found.
kpeter@2573
   324
    bool run(bool min_mean_cc = false) {
kpeter@2573
   325
      return init() && start(min_mean_cc);
kpeter@2556
   326
    }
kpeter@2556
   327
kpeter@2581
   328
    /// @}
kpeter@2581
   329
kpeter@2581
   330
    /// \name Query Functions
kpeter@2581
   331
    /// The result of the algorithm can be obtained using these
kpeter@2620
   332
    /// functions.\n
kpeter@2620
   333
    /// \ref lemon::CycleCanceling::run() "run()" must be called before
kpeter@2620
   334
    /// using them.
kpeter@2581
   335
kpeter@2581
   336
    /// @{
kpeter@2581
   337
kpeter@2620
   338
    /// \brief Return a const reference to the edge map storing the
kpeter@2573
   339
    /// found flow.
deba@2440
   340
    ///
kpeter@2620
   341
    /// Return a const reference to the edge map storing the found flow.
deba@2440
   342
    ///
deba@2440
   343
    /// \pre \ref run() must be called before using this function.
deba@2440
   344
    const FlowMap& flowMap() const {
kpeter@2581
   345
      return *_flow;
kpeter@2581
   346
    }
kpeter@2581
   347
kpeter@2620
   348
    /// \brief Return a const reference to the node map storing the
kpeter@2581
   349
    /// found potentials (the dual solution).
kpeter@2581
   350
    ///
kpeter@2620
   351
    /// Return a const reference to the node map storing the found
kpeter@2581
   352
    /// potentials (the dual solution).
kpeter@2581
   353
    ///
kpeter@2581
   354
    /// \pre \ref run() must be called before using this function.
kpeter@2581
   355
    const PotentialMap& potentialMap() const {
kpeter@2581
   356
      return *_potential;
kpeter@2581
   357
    }
kpeter@2581
   358
kpeter@2620
   359
    /// \brief Return the flow on the given edge.
kpeter@2581
   360
    ///
kpeter@2620
   361
    /// Return the flow on the given edge.
kpeter@2581
   362
    ///
kpeter@2581
   363
    /// \pre \ref run() must be called before using this function.
kpeter@2581
   364
    Capacity flow(const Edge& edge) const {
kpeter@2581
   365
      return (*_flow)[edge];
kpeter@2581
   366
    }
kpeter@2581
   367
kpeter@2620
   368
    /// \brief Return the potential of the given node.
kpeter@2581
   369
    ///
kpeter@2620
   370
    /// Return the potential of the given node.
kpeter@2581
   371
    ///
kpeter@2581
   372
    /// \pre \ref run() must be called before using this function.
kpeter@2581
   373
    Cost potential(const Node& node) const {
kpeter@2581
   374
      return (*_potential)[node];
deba@2440
   375
    }
deba@2440
   376
kpeter@2620
   377
    /// \brief Return the total cost of the found flow.
deba@2440
   378
    ///
kpeter@2620
   379
    /// Return the total cost of the found flow. The complexity of the
deba@2440
   380
    /// function is \f$ O(e) \f$.
deba@2440
   381
    ///
deba@2440
   382
    /// \pre \ref run() must be called before using this function.
deba@2440
   383
    Cost totalCost() const {
deba@2440
   384
      Cost c = 0;
kpeter@2573
   385
      for (EdgeIt e(_graph); e != INVALID; ++e)
kpeter@2581
   386
        c += (*_flow)[e] * _cost[e];
deba@2440
   387
      return c;
deba@2440
   388
    }
deba@2440
   389
kpeter@2581
   390
    /// @}
kpeter@2581
   391
kpeter@2573
   392
  private:
deba@2440
   393
kpeter@2620
   394
    /// Initialize the algorithm.
deba@2440
   395
    bool init() {
kpeter@2573
   396
      if (!_valid_supply) return false;
deba@2440
   397
kpeter@2581
   398
      // Initializing flow and potential maps
kpeter@2581
   399
      if (!_flow) {
kpeter@2581
   400
        _flow = new FlowMap(_graph);
kpeter@2581
   401
        _local_flow = true;
kpeter@2581
   402
      }
kpeter@2581
   403
      if (!_potential) {
kpeter@2581
   404
        _potential = new PotentialMap(_graph);
kpeter@2581
   405
        _local_potential = true;
kpeter@2581
   406
      }
kpeter@2581
   407
kpeter@2581
   408
      _res_graph = new ResGraph(_graph, _capacity, *_flow);
kpeter@2581
   409
kpeter@2573
   410
      // Finding a feasible flow using Circulation
kpeter@2556
   411
      Circulation< Graph, ConstMap<Edge, Capacity>, CapacityEdgeMap,
kpeter@2556
   412
                   SupplyMap >
kpeter@2581
   413
        circulation( _graph, constMap<Edge>(Capacity(0)), _capacity,
kpeter@2573
   414
                     _supply );
kpeter@2581
   415
      return circulation.flowMap(*_flow).run();
deba@2440
   416
    }
deba@2440
   417
kpeter@2573
   418
    bool start(bool min_mean_cc) {
kpeter@2573
   419
      if (min_mean_cc)
kpeter@2581
   420
        startMinMean();
kpeter@2573
   421
      else
kpeter@2581
   422
        start();
kpeter@2581
   423
kpeter@2581
   424
      // Handling non-zero lower bounds
kpeter@2581
   425
      if (_lower) {
kpeter@2581
   426
        for (EdgeIt e(_graph); e != INVALID; ++e)
kpeter@2581
   427
          (*_flow)[e] += (*_lower)[e];
kpeter@2581
   428
      }
kpeter@2581
   429
      return true;
kpeter@2573
   430
    }
kpeter@2573
   431
kpeter@2620
   432
    /// \brief Execute the algorithm using \ref BellmanFord.
kpeter@2573
   433
    ///
kpeter@2620
   434
    /// Execute the algorithm using the \ref BellmanFord
kpeter@2573
   435
    /// "Bellman-Ford" algorithm for negative cycle detection with
kpeter@2573
   436
    /// successively larger limit for the number of iterations.
kpeter@2581
   437
    void start() {
kpeter@2581
   438
      typename BellmanFord<ResGraph, ResidualCostMap>::PredMap pred(*_res_graph);
kpeter@2581
   439
      typename ResGraph::template NodeMap<int> visited(*_res_graph);
deba@2440
   440
      std::vector<ResEdge> cycle;
kpeter@2573
   441
      int node_num = countNodes(_graph);
deba@2440
   442
kpeter@2573
   443
      int length_bound = BF_FIRST_LIMIT;
deba@2440
   444
      bool optimal = false;
deba@2440
   445
      while (!optimal) {
kpeter@2581
   446
        BellmanFord<ResGraph, ResidualCostMap> bf(*_res_graph, _res_cost);
kpeter@2556
   447
        bf.predMap(pred);
kpeter@2556
   448
        bf.init(0);
kpeter@2556
   449
        int iter_num = 0;
kpeter@2556
   450
        bool cycle_found = false;
kpeter@2556
   451
        while (!cycle_found) {
kpeter@2556
   452
          int curr_iter_num = iter_num + length_bound <= node_num ?
kpeter@2556
   453
                              length_bound : node_num - iter_num;
kpeter@2556
   454
          iter_num += curr_iter_num;
kpeter@2556
   455
          int real_iter_num = curr_iter_num;
kpeter@2556
   456
          for (int i = 0; i < curr_iter_num; ++i) {
kpeter@2556
   457
            if (bf.processNextWeakRound()) {
kpeter@2556
   458
              real_iter_num = i;
kpeter@2556
   459
              break;
kpeter@2556
   460
            }
kpeter@2556
   461
          }
kpeter@2556
   462
          if (real_iter_num < curr_iter_num) {
kpeter@2581
   463
            // Optimal flow is found
kpeter@2556
   464
            optimal = true;
kpeter@2581
   465
            // Setting node potentials
kpeter@2581
   466
            for (NodeIt n(_graph); n != INVALID; ++n)
kpeter@2581
   467
              (*_potential)[n] = bf.dist(n);
kpeter@2556
   468
            break;
kpeter@2556
   469
          } else {
kpeter@2556
   470
            // Searching for node disjoint negative cycles
kpeter@2581
   471
            for (ResNodeIt n(*_res_graph); n != INVALID; ++n)
kpeter@2556
   472
              visited[n] = 0;
kpeter@2556
   473
            int id = 0;
kpeter@2581
   474
            for (ResNodeIt n(*_res_graph); n != INVALID; ++n) {
kpeter@2556
   475
              if (visited[n] > 0) continue;
kpeter@2556
   476
              visited[n] = ++id;
kpeter@2556
   477
              ResNode u = pred[n] == INVALID ?
kpeter@2581
   478
                          INVALID : _res_graph->source(pred[n]);
kpeter@2556
   479
              while (u != INVALID && visited[u] == 0) {
kpeter@2556
   480
                visited[u] = id;
kpeter@2556
   481
                u = pred[u] == INVALID ?
kpeter@2581
   482
                    INVALID : _res_graph->source(pred[u]);
kpeter@2556
   483
              }
kpeter@2556
   484
              if (u != INVALID && visited[u] == id) {
kpeter@2556
   485
                // Finding the negative cycle
kpeter@2556
   486
                cycle_found = true;
kpeter@2556
   487
                cycle.clear();
kpeter@2556
   488
                ResEdge e = pred[u];
kpeter@2556
   489
                cycle.push_back(e);
kpeter@2581
   490
                Capacity d = _res_graph->rescap(e);
kpeter@2581
   491
                while (_res_graph->source(e) != u) {
kpeter@2581
   492
                  cycle.push_back(e = pred[_res_graph->source(e)]);
kpeter@2581
   493
                  if (_res_graph->rescap(e) < d)
kpeter@2581
   494
                    d = _res_graph->rescap(e);
kpeter@2556
   495
                }
kpeter@2573
   496
kpeter@2556
   497
                // Augmenting along the cycle
kpeter@2573
   498
                for (int i = 0; i < int(cycle.size()); ++i)
kpeter@2581
   499
                  _res_graph->augment(cycle[i], d);
kpeter@2556
   500
              }
kpeter@2556
   501
            }
kpeter@2556
   502
          }
deba@2440
   503
kpeter@2556
   504
          if (!cycle_found)
kpeter@2593
   505
            length_bound = length_bound * BF_LIMIT_FACTOR / 100;
kpeter@2556
   506
        }
deba@2440
   507
      }
deba@2440
   508
    }
deba@2440
   509
kpeter@2620
   510
    /// \brief Execute the algorithm using \ref MinMeanCycle.
kpeter@2573
   511
    ///
kpeter@2620
   512
    /// Execute the algorithm using \ref MinMeanCycle for negative
kpeter@2573
   513
    /// cycle detection.
kpeter@2581
   514
    void startMinMean() {
deba@2440
   515
      typedef Path<ResGraph> ResPath;
kpeter@2581
   516
      MinMeanCycle<ResGraph, ResidualCostMap> mmc(*_res_graph, _res_cost);
deba@2440
   517
      ResPath cycle;
deba@2440
   518
deba@2440
   519
      mmc.cyclePath(cycle).init();
deba@2440
   520
      if (mmc.findMinMean()) {
kpeter@2556
   521
        while (mmc.cycleLength() < 0) {
kpeter@2556
   522
          // Finding the cycle
kpeter@2556
   523
          mmc.findCycle();
deba@2440
   524
kpeter@2556
   525
          // Finding the largest flow amount that can be augmented
kpeter@2556
   526
          // along the cycle
kpeter@2556
   527
          Capacity delta = 0;
kpeter@2556
   528
          for (typename ResPath::EdgeIt e(cycle); e != INVALID; ++e) {
kpeter@2581
   529
            if (delta == 0 || _res_graph->rescap(e) < delta)
kpeter@2581
   530
              delta = _res_graph->rescap(e);
kpeter@2556
   531
          }
deba@2440
   532
kpeter@2556
   533
          // Augmenting along the cycle
kpeter@2556
   534
          for (typename ResPath::EdgeIt e(cycle); e != INVALID; ++e)
kpeter@2581
   535
            _res_graph->augment(e, delta);
deba@2440
   536
kpeter@2556
   537
          // Finding the minimum cycle mean for the modified residual
kpeter@2556
   538
          // graph
kpeter@2556
   539
          mmc.reset();
kpeter@2556
   540
          if (!mmc.findMinMean()) break;
kpeter@2556
   541
        }
deba@2440
   542
      }
deba@2440
   543
kpeter@2581
   544
      // Computing node potentials
kpeter@2581
   545
      BellmanFord<ResGraph, ResidualCostMap> bf(*_res_graph, _res_cost);
kpeter@2581
   546
      bf.init(0); bf.start();
kpeter@2581
   547
      for (NodeIt n(_graph); n != INVALID; ++n)
kpeter@2581
   548
        (*_potential)[n] = bf.dist(n);
deba@2440
   549
    }
deba@2440
   550
deba@2440
   551
  }; //class CycleCanceling
deba@2440
   552
deba@2440
   553
  ///@}
deba@2440
   554
deba@2440
   555
} //namespace lemon
deba@2440
   556
deba@2440
   557
#endif //LEMON_CYCLE_CANCELING_H