lemon/grosso_locatelli_pullan_mc.h
author Alpar Juttner <alpar@cs.elte.hu>
Mon, 25 Oct 2010 15:33:57 +0200
changeset 1015 481496e6d71f
child 1022 8583fb74238c
permissions -rw-r--r--
SOURCE_BROWSER Doxygen switch is configurable from CMAKE (#395)
kpeter@999
     1
/* -*- mode: C++; indent-tabs-mode: nil; -*-
kpeter@999
     2
 *
kpeter@999
     3
 * This file is a part of LEMON, a generic C++ optimization library.
kpeter@999
     4
 *
kpeter@999
     5
 * Copyright (C) 2003-2010
kpeter@999
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
kpeter@999
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
kpeter@999
     8
 *
kpeter@999
     9
 * Permission to use, modify and distribute this software is granted
kpeter@999
    10
 * provided that this copyright notice appears in all copies. For
kpeter@999
    11
 * precise terms see the accompanying LICENSE file.
kpeter@999
    12
 *
kpeter@999
    13
 * This software is provided "AS IS" with no warranty of any kind,
kpeter@999
    14
 * express or implied, and with no claim as to its suitability for any
kpeter@999
    15
 * purpose.
kpeter@999
    16
 *
kpeter@999
    17
 */
kpeter@999
    18
kpeter@999
    19
#ifndef LEMON_GROSSO_LOCATELLI_PULLAN_MC_H
kpeter@999
    20
#define LEMON_GROSSO_LOCATELLI_PULLAN_MC_H
kpeter@999
    21
kpeter@999
    22
/// \ingroup approx_algs
kpeter@999
    23
///
kpeter@999
    24
/// \file
kpeter@999
    25
/// \brief The iterated local search algorithm of Grosso, Locatelli, and Pullan
kpeter@999
    26
/// for the maximum clique problem
kpeter@999
    27
kpeter@999
    28
#include <vector>
kpeter@999
    29
#include <limits>
kpeter@999
    30
#include <lemon/core.h>
kpeter@999
    31
#include <lemon/random.h>
kpeter@999
    32
kpeter@999
    33
namespace lemon {
kpeter@999
    34
kpeter@999
    35
  /// \addtogroup approx_algs
kpeter@999
    36
  /// @{
kpeter@999
    37
kpeter@999
    38
  /// \brief Implementation of the iterated local search algorithm of Grosso,
kpeter@999
    39
  /// Locatelli, and Pullan for the maximum clique problem
kpeter@999
    40
  ///
kpeter@999
    41
  /// \ref GrossoLocatelliPullanMc implements the iterated local search
kpeter@999
    42
  /// algorithm of Grosso, Locatelli, and Pullan for solving the \e maximum
kpeter@999
    43
  /// \e clique \e problem \ref grosso08maxclique.
kpeter@999
    44
  /// It is to find the largest complete subgraph (\e clique) in an
kpeter@999
    45
  /// undirected graph, i.e., the largest set of nodes where each
kpeter@999
    46
  /// pair of nodes is connected.
kpeter@999
    47
  ///
kpeter@999
    48
  /// This class provides a simple but highly efficient and robust heuristic
kpeter@999
    49
  /// method that quickly finds a large clique, but not necessarily the
kpeter@999
    50
  /// largest one.
kpeter@999
    51
  ///
kpeter@999
    52
  /// \tparam GR The undirected graph type the algorithm runs on.
kpeter@999
    53
  ///
kpeter@999
    54
  /// \note %GrossoLocatelliPullanMc provides three different node selection
kpeter@999
    55
  /// rules, from which the most powerful one is used by default.
kpeter@999
    56
  /// For more information, see \ref SelectionRule.
kpeter@999
    57
  template <typename GR>
kpeter@999
    58
  class GrossoLocatelliPullanMc
kpeter@999
    59
  {
kpeter@999
    60
  public:
kpeter@999
    61
kpeter@999
    62
    /// \brief Constants for specifying the node selection rule.
kpeter@999
    63
    ///
kpeter@999
    64
    /// Enum type containing constants for specifying the node selection rule
kpeter@999
    65
    /// for the \ref run() function.
kpeter@999
    66
    ///
kpeter@999
    67
    /// During the algorithm, nodes are selected for addition to the current
kpeter@999
    68
    /// clique according to the applied rule.
kpeter@999
    69
    /// In general, the PENALTY_BASED rule turned out to be the most powerful
kpeter@999
    70
    /// and the most robust, thus it is the default option.
kpeter@999
    71
    /// However, another selection rule can be specified using the \ref run()
kpeter@999
    72
    /// function with the proper parameter.
kpeter@999
    73
    enum SelectionRule {
kpeter@999
    74
kpeter@999
    75
      /// A node is selected randomly without any evaluation at each step.
kpeter@999
    76
      RANDOM,
kpeter@999
    77
kpeter@999
    78
      /// A node of maximum degree is selected randomly at each step.
kpeter@999
    79
      DEGREE_BASED,
kpeter@999
    80
kpeter@999
    81
      /// A node of minimum penalty is selected randomly at each step.
kpeter@999
    82
      /// The node penalties are updated adaptively after each stage of the
kpeter@999
    83
      /// search process.
kpeter@999
    84
      PENALTY_BASED
kpeter@999
    85
    };
kpeter@999
    86
kpeter@999
    87
  private:
kpeter@999
    88
kpeter@999
    89
    TEMPLATE_GRAPH_TYPEDEFS(GR);
kpeter@999
    90
kpeter@999
    91
    typedef std::vector<int> IntVector;
kpeter@999
    92
    typedef std::vector<char> BoolVector;
kpeter@999
    93
    typedef std::vector<BoolVector> BoolMatrix;
kpeter@999
    94
    // Note: vector<char> is used instead of vector<bool> for efficiency reasons
kpeter@999
    95
kpeter@999
    96
    const GR &_graph;
kpeter@999
    97
    IntNodeMap _id;
kpeter@999
    98
kpeter@999
    99
    // Internal matrix representation of the graph
kpeter@999
   100
    BoolMatrix _gr;
kpeter@999
   101
    int _n;
kpeter@999
   102
kpeter@999
   103
    // The current clique
kpeter@999
   104
    BoolVector _clique;
kpeter@999
   105
    int _size;
kpeter@999
   106
kpeter@999
   107
    // The best clique found so far
kpeter@999
   108
    BoolVector _best_clique;
kpeter@999
   109
    int _best_size;
kpeter@999
   110
kpeter@999
   111
    // The "distances" of the nodes from the current clique.
kpeter@999
   112
    // _delta[u] is the number of nodes in the clique that are
kpeter@999
   113
    // not connected with u.
kpeter@999
   114
    IntVector _delta;
kpeter@999
   115
kpeter@999
   116
    // The current tabu set
kpeter@999
   117
    BoolVector _tabu;
kpeter@999
   118
kpeter@999
   119
    // Random number generator
kpeter@999
   120
    Random _rnd;
kpeter@999
   121
kpeter@999
   122
  private:
kpeter@999
   123
kpeter@999
   124
    // Implementation of the RANDOM node selection rule.
kpeter@999
   125
    class RandomSelectionRule
kpeter@999
   126
    {
kpeter@999
   127
    private:
kpeter@999
   128
kpeter@999
   129
      // References to the algorithm instance
kpeter@999
   130
      const BoolVector &_clique;
kpeter@999
   131
      const IntVector  &_delta;
kpeter@999
   132
      const BoolVector &_tabu;
kpeter@999
   133
      Random &_rnd;
kpeter@999
   134
kpeter@999
   135
      // Pivot rule data
kpeter@999
   136
      int _n;
kpeter@999
   137
kpeter@999
   138
    public:
kpeter@999
   139
kpeter@999
   140
      // Constructor
kpeter@999
   141
      RandomSelectionRule(GrossoLocatelliPullanMc &mc) :
kpeter@999
   142
        _clique(mc._clique), _delta(mc._delta), _tabu(mc._tabu),
kpeter@999
   143
        _rnd(mc._rnd), _n(mc._n)
kpeter@999
   144
      {}
kpeter@999
   145
kpeter@999
   146
      // Return a node index for a feasible add move or -1 if no one exists
kpeter@999
   147
      int nextFeasibleAddNode() const {
kpeter@999
   148
        int start_node = _rnd[_n];
kpeter@999
   149
        for (int i = start_node; i != _n; i++) {
kpeter@999
   150
          if (_delta[i] == 0 && !_tabu[i]) return i;
kpeter@999
   151
        }
kpeter@999
   152
        for (int i = 0; i != start_node; i++) {
kpeter@999
   153
          if (_delta[i] == 0 && !_tabu[i]) return i;
kpeter@999
   154
        }
kpeter@999
   155
        return -1;
kpeter@999
   156
      }
kpeter@999
   157
kpeter@999
   158
      // Return a node index for a feasible swap move or -1 if no one exists
kpeter@999
   159
      int nextFeasibleSwapNode() const {
kpeter@999
   160
        int start_node = _rnd[_n];
kpeter@999
   161
        for (int i = start_node; i != _n; i++) {
kpeter@999
   162
          if (!_clique[i] && _delta[i] == 1 && !_tabu[i]) return i;
kpeter@999
   163
        }
kpeter@999
   164
        for (int i = 0; i != start_node; i++) {
kpeter@999
   165
          if (!_clique[i] && _delta[i] == 1 && !_tabu[i]) return i;
kpeter@999
   166
        }
kpeter@999
   167
        return -1;
kpeter@999
   168
      }
kpeter@999
   169
kpeter@999
   170
      // Return a node index for an add move or -1 if no one exists
kpeter@999
   171
      int nextAddNode() const {
kpeter@999
   172
        int start_node = _rnd[_n];
kpeter@999
   173
        for (int i = start_node; i != _n; i++) {
kpeter@999
   174
          if (_delta[i] == 0) return i;
kpeter@999
   175
        }
kpeter@999
   176
        for (int i = 0; i != start_node; i++) {
kpeter@999
   177
          if (_delta[i] == 0) return i;
kpeter@999
   178
        }
kpeter@999
   179
        return -1;
kpeter@999
   180
      }
kpeter@999
   181
kpeter@999
   182
      // Update internal data structures between stages (if necessary)
kpeter@999
   183
      void update() {}
kpeter@999
   184
kpeter@999
   185
    }; //class RandomSelectionRule
kpeter@999
   186
kpeter@999
   187
kpeter@999
   188
    // Implementation of the DEGREE_BASED node selection rule.
kpeter@999
   189
    class DegreeBasedSelectionRule
kpeter@999
   190
    {
kpeter@999
   191
    private:
kpeter@999
   192
kpeter@999
   193
      // References to the algorithm instance
kpeter@999
   194
      const BoolVector &_clique;
kpeter@999
   195
      const IntVector  &_delta;
kpeter@999
   196
      const BoolVector &_tabu;
kpeter@999
   197
      Random &_rnd;
kpeter@999
   198
kpeter@999
   199
      // Pivot rule data
kpeter@999
   200
      int _n;
kpeter@999
   201
      IntVector _deg;
kpeter@999
   202
kpeter@999
   203
    public:
kpeter@999
   204
kpeter@999
   205
      // Constructor
kpeter@999
   206
      DegreeBasedSelectionRule(GrossoLocatelliPullanMc &mc) :
kpeter@999
   207
        _clique(mc._clique), _delta(mc._delta), _tabu(mc._tabu),
kpeter@999
   208
        _rnd(mc._rnd), _n(mc._n), _deg(_n)
kpeter@999
   209
      {
kpeter@999
   210
        for (int i = 0; i != _n; i++) {
kpeter@999
   211
          int d = 0;
kpeter@999
   212
          BoolVector &row = mc._gr[i];
kpeter@999
   213
          for (int j = 0; j != _n; j++) {
kpeter@999
   214
            if (row[j]) d++;
kpeter@999
   215
          }
kpeter@999
   216
          _deg[i] = d;
kpeter@999
   217
        }
kpeter@999
   218
      }
kpeter@999
   219
kpeter@999
   220
      // Return a node index for a feasible add move or -1 if no one exists
kpeter@999
   221
      int nextFeasibleAddNode() const {
kpeter@999
   222
        int start_node = _rnd[_n];
kpeter@999
   223
        int node = -1, max_deg = -1;
kpeter@999
   224
        for (int i = start_node; i != _n; i++) {
kpeter@999
   225
          if (_delta[i] == 0 && !_tabu[i] && _deg[i] > max_deg) {
kpeter@999
   226
            node = i;
kpeter@999
   227
            max_deg = _deg[i];
kpeter@999
   228
          }
kpeter@999
   229
        }
kpeter@999
   230
        for (int i = 0; i != start_node; i++) {
kpeter@999
   231
          if (_delta[i] == 0 && !_tabu[i] && _deg[i] > max_deg) {
kpeter@999
   232
            node = i;
kpeter@999
   233
            max_deg = _deg[i];
kpeter@999
   234
          }
kpeter@999
   235
        }
kpeter@999
   236
        return node;
kpeter@999
   237
      }
kpeter@999
   238
kpeter@999
   239
      // Return a node index for a feasible swap move or -1 if no one exists
kpeter@999
   240
      int nextFeasibleSwapNode() const {
kpeter@999
   241
        int start_node = _rnd[_n];
kpeter@999
   242
        int node = -1, max_deg = -1;
kpeter@999
   243
        for (int i = start_node; i != _n; i++) {
kpeter@999
   244
          if (!_clique[i] && _delta[i] == 1 && !_tabu[i] &&
kpeter@999
   245
              _deg[i] > max_deg) {
kpeter@999
   246
            node = i;
kpeter@999
   247
            max_deg = _deg[i];
kpeter@999
   248
          }
kpeter@999
   249
        }
kpeter@999
   250
        for (int i = 0; i != start_node; i++) {
kpeter@999
   251
          if (!_clique[i] && _delta[i] == 1 && !_tabu[i] &&
kpeter@999
   252
              _deg[i] > max_deg) {
kpeter@999
   253
            node = i;
kpeter@999
   254
            max_deg = _deg[i];
kpeter@999
   255
          }
kpeter@999
   256
        }
kpeter@999
   257
        return node;
kpeter@999
   258
      }
kpeter@999
   259
kpeter@999
   260
      // Return a node index for an add move or -1 if no one exists
kpeter@999
   261
      int nextAddNode() const {
kpeter@999
   262
        int start_node = _rnd[_n];
kpeter@999
   263
        int node = -1, max_deg = -1;
kpeter@999
   264
        for (int i = start_node; i != _n; i++) {
kpeter@999
   265
          if (_delta[i] == 0 && _deg[i] > max_deg) {
kpeter@999
   266
            node = i;
kpeter@999
   267
            max_deg = _deg[i];
kpeter@999
   268
          }
kpeter@999
   269
        }
kpeter@999
   270
        for (int i = 0; i != start_node; i++) {
kpeter@999
   271
          if (_delta[i] == 0 && _deg[i] > max_deg) {
kpeter@999
   272
            node = i;
kpeter@999
   273
            max_deg = _deg[i];
kpeter@999
   274
          }
kpeter@999
   275
        }
kpeter@999
   276
        return node;
kpeter@999
   277
      }
kpeter@999
   278
kpeter@999
   279
      // Update internal data structures between stages (if necessary)
kpeter@999
   280
      void update() {}
kpeter@999
   281
kpeter@999
   282
    }; //class DegreeBasedSelectionRule
kpeter@999
   283
kpeter@999
   284
kpeter@999
   285
    // Implementation of the PENALTY_BASED node selection rule.
kpeter@999
   286
    class PenaltyBasedSelectionRule
kpeter@999
   287
    {
kpeter@999
   288
    private:
kpeter@999
   289
kpeter@999
   290
      // References to the algorithm instance
kpeter@999
   291
      const BoolVector &_clique;
kpeter@999
   292
      const IntVector  &_delta;
kpeter@999
   293
      const BoolVector &_tabu;
kpeter@999
   294
      Random &_rnd;
kpeter@999
   295
kpeter@999
   296
      // Pivot rule data
kpeter@999
   297
      int _n;
kpeter@999
   298
      IntVector _penalty;
kpeter@999
   299
kpeter@999
   300
    public:
kpeter@999
   301
kpeter@999
   302
      // Constructor
kpeter@999
   303
      PenaltyBasedSelectionRule(GrossoLocatelliPullanMc &mc) :
kpeter@999
   304
        _clique(mc._clique), _delta(mc._delta), _tabu(mc._tabu),
kpeter@999
   305
        _rnd(mc._rnd), _n(mc._n), _penalty(_n, 0)
kpeter@999
   306
      {}
kpeter@999
   307
kpeter@999
   308
      // Return a node index for a feasible add move or -1 if no one exists
kpeter@999
   309
      int nextFeasibleAddNode() const {
kpeter@999
   310
        int start_node = _rnd[_n];
kpeter@999
   311
        int node = -1, min_p = std::numeric_limits<int>::max();
kpeter@999
   312
        for (int i = start_node; i != _n; i++) {
kpeter@999
   313
          if (_delta[i] == 0 && !_tabu[i] && _penalty[i] < min_p) {
kpeter@999
   314
            node = i;
kpeter@999
   315
            min_p = _penalty[i];
kpeter@999
   316
          }
kpeter@999
   317
        }
kpeter@999
   318
        for (int i = 0; i != start_node; i++) {
kpeter@999
   319
          if (_delta[i] == 0 && !_tabu[i] && _penalty[i] < min_p) {
kpeter@999
   320
            node = i;
kpeter@999
   321
            min_p = _penalty[i];
kpeter@999
   322
          }
kpeter@999
   323
        }
kpeter@999
   324
        return node;
kpeter@999
   325
      }
kpeter@999
   326
kpeter@999
   327
      // Return a node index for a feasible swap move or -1 if no one exists
kpeter@999
   328
      int nextFeasibleSwapNode() const {
kpeter@999
   329
        int start_node = _rnd[_n];
kpeter@999
   330
        int node = -1, min_p = std::numeric_limits<int>::max();
kpeter@999
   331
        for (int i = start_node; i != _n; i++) {
kpeter@999
   332
          if (!_clique[i] && _delta[i] == 1 && !_tabu[i] &&
kpeter@999
   333
              _penalty[i] < min_p) {
kpeter@999
   334
            node = i;
kpeter@999
   335
            min_p = _penalty[i];
kpeter@999
   336
          }
kpeter@999
   337
        }
kpeter@999
   338
        for (int i = 0; i != start_node; i++) {
kpeter@999
   339
          if (!_clique[i] && _delta[i] == 1 && !_tabu[i] &&
kpeter@999
   340
              _penalty[i] < min_p) {
kpeter@999
   341
            node = i;
kpeter@999
   342
            min_p = _penalty[i];
kpeter@999
   343
          }
kpeter@999
   344
        }
kpeter@999
   345
        return node;
kpeter@999
   346
      }
kpeter@999
   347
kpeter@999
   348
      // Return a node index for an add move or -1 if no one exists
kpeter@999
   349
      int nextAddNode() const {
kpeter@999
   350
        int start_node = _rnd[_n];
kpeter@999
   351
        int node = -1, min_p = std::numeric_limits<int>::max();
kpeter@999
   352
        for (int i = start_node; i != _n; i++) {
kpeter@999
   353
          if (_delta[i] == 0 && _penalty[i] < min_p) {
kpeter@999
   354
            node = i;
kpeter@999
   355
            min_p = _penalty[i];
kpeter@999
   356
          }
kpeter@999
   357
        }
kpeter@999
   358
        for (int i = 0; i != start_node; i++) {
kpeter@999
   359
          if (_delta[i] == 0 && _penalty[i] < min_p) {
kpeter@999
   360
            node = i;
kpeter@999
   361
            min_p = _penalty[i];
kpeter@999
   362
          }
kpeter@999
   363
        }
kpeter@999
   364
        return node;
kpeter@999
   365
      }
kpeter@999
   366
kpeter@999
   367
      // Update internal data structures between stages (if necessary)
kpeter@999
   368
      void update() {}
kpeter@999
   369
kpeter@999
   370
    }; //class PenaltyBasedSelectionRule
kpeter@999
   371
kpeter@999
   372
  public:
kpeter@999
   373
kpeter@999
   374
    /// \brief Constructor.
kpeter@999
   375
    ///
kpeter@999
   376
    /// Constructor.
kpeter@999
   377
    /// The global \ref rnd "random number generator instance" is used
kpeter@999
   378
    /// during the algorithm.
kpeter@999
   379
    ///
kpeter@999
   380
    /// \param graph The undirected graph the algorithm runs on.
kpeter@999
   381
    GrossoLocatelliPullanMc(const GR& graph) :
kpeter@999
   382
      _graph(graph), _id(_graph), _rnd(rnd)
kpeter@999
   383
    {}
kpeter@999
   384
kpeter@999
   385
    /// \brief Constructor with random seed.
kpeter@999
   386
    ///
kpeter@999
   387
    /// Constructor with random seed.
kpeter@999
   388
    ///
kpeter@999
   389
    /// \param graph The undirected graph the algorithm runs on.
kpeter@999
   390
    /// \param seed Seed value for the internal random number generator
kpeter@999
   391
    /// that is used during the algorithm.
kpeter@999
   392
    GrossoLocatelliPullanMc(const GR& graph, int seed) :
kpeter@999
   393
      _graph(graph), _id(_graph), _rnd(seed)
kpeter@999
   394
    {}
kpeter@999
   395
kpeter@999
   396
    /// \brief Constructor with random number generator.
kpeter@999
   397
    ///
kpeter@999
   398
    /// Constructor with random number generator.
kpeter@999
   399
    ///
kpeter@999
   400
    /// \param graph The undirected graph the algorithm runs on.
kpeter@999
   401
    /// \param random A random number generator that is used during the
kpeter@999
   402
    /// algorithm.
kpeter@999
   403
    GrossoLocatelliPullanMc(const GR& graph, const Random& random) :
kpeter@999
   404
      _graph(graph), _id(_graph), _rnd(random)
kpeter@999
   405
    {}
kpeter@999
   406
kpeter@999
   407
    /// \name Execution Control
kpeter@999
   408
    /// @{
kpeter@999
   409
kpeter@999
   410
    /// \brief Runs the algorithm.
kpeter@999
   411
    ///
kpeter@999
   412
    /// This function runs the algorithm.
kpeter@999
   413
    ///
kpeter@999
   414
    /// \param step_num The maximum number of node selections (steps)
kpeter@999
   415
    /// during the search process.
kpeter@999
   416
    /// This parameter controls the running time and the success of the
kpeter@999
   417
    /// algorithm. For larger values, the algorithm runs slower but it more
kpeter@999
   418
    /// likely finds larger cliques. For smaller values, the algorithm is
kpeter@999
   419
    /// faster but probably gives worse results.
kpeter@999
   420
    /// \param rule The node selection rule. For more information, see
kpeter@999
   421
    /// \ref SelectionRule.
kpeter@999
   422
    ///
kpeter@999
   423
    /// \return The size of the found clique.
kpeter@999
   424
    int run(int step_num = 100000,
kpeter@999
   425
            SelectionRule rule = PENALTY_BASED)
kpeter@999
   426
    {
kpeter@999
   427
      init();
kpeter@999
   428
      switch (rule) {
kpeter@999
   429
        case RANDOM:
kpeter@999
   430
          return start<RandomSelectionRule>(step_num);
kpeter@999
   431
        case DEGREE_BASED:
kpeter@999
   432
          return start<DegreeBasedSelectionRule>(step_num);
kpeter@999
   433
        case PENALTY_BASED:
kpeter@999
   434
          return start<PenaltyBasedSelectionRule>(step_num);
kpeter@999
   435
      }
kpeter@999
   436
      return 0; // avoid warning
kpeter@999
   437
    }
kpeter@999
   438
kpeter@999
   439
    /// @}
kpeter@999
   440
kpeter@999
   441
    /// \name Query Functions
kpeter@999
   442
    /// @{
kpeter@999
   443
kpeter@999
   444
    /// \brief The size of the found clique
kpeter@999
   445
    ///
kpeter@999
   446
    /// This function returns the size of the found clique.
kpeter@999
   447
    ///
kpeter@999
   448
    /// \pre run() must be called before using this function.
kpeter@999
   449
    int cliqueSize() const {
kpeter@999
   450
      return _best_size;
kpeter@999
   451
    }
kpeter@999
   452
kpeter@999
   453
    /// \brief Gives back the found clique in a \c bool node map
kpeter@999
   454
    ///
kpeter@999
   455
    /// This function gives back the characteristic vector of the found
kpeter@999
   456
    /// clique in the given node map.
kpeter@999
   457
    /// It must be a \ref concepts::WriteMap "writable" node map with
kpeter@999
   458
    /// \c bool (or convertible) value type.
kpeter@999
   459
    ///
kpeter@999
   460
    /// \pre run() must be called before using this function.
kpeter@999
   461
    template <typename CliqueMap>
kpeter@999
   462
    void cliqueMap(CliqueMap &map) const {
kpeter@999
   463
      for (NodeIt n(_graph); n != INVALID; ++n) {
kpeter@999
   464
        map[n] = static_cast<bool>(_best_clique[_id[n]]);
kpeter@999
   465
      }
kpeter@999
   466
    }
kpeter@999
   467
kpeter@999
   468
    /// \brief Iterator to list the nodes of the found clique
kpeter@999
   469
    ///
kpeter@999
   470
    /// This iterator class lists the nodes of the found clique.
kpeter@999
   471
    /// Before using it, you must allocate a GrossoLocatelliPullanMc instance
kpeter@999
   472
    /// and call its \ref GrossoLocatelliPullanMc::run() "run()" method.
kpeter@999
   473
    ///
kpeter@999
   474
    /// The following example prints out the IDs of the nodes in the found
kpeter@999
   475
    /// clique.
kpeter@999
   476
    /// \code
kpeter@999
   477
    ///   GrossoLocatelliPullanMc<Graph> mc(g);
kpeter@999
   478
    ///   mc.run();
kpeter@999
   479
    ///   for (GrossoLocatelliPullanMc<Graph>::CliqueNodeIt n(mc);
kpeter@999
   480
    ///        n != INVALID; ++n)
kpeter@999
   481
    ///   {
kpeter@999
   482
    ///     std::cout << g.id(n) << std::endl;
kpeter@999
   483
    ///   }
kpeter@999
   484
    /// \endcode
kpeter@999
   485
    class CliqueNodeIt
kpeter@999
   486
    {
kpeter@999
   487
    private:
kpeter@999
   488
      NodeIt _it;
kpeter@999
   489
      BoolNodeMap _map;
kpeter@999
   490
kpeter@999
   491
    public:
kpeter@999
   492
kpeter@999
   493
      /// Constructor
kpeter@999
   494
kpeter@999
   495
      /// Constructor.
kpeter@999
   496
      /// \param mc The algorithm instance.
kpeter@999
   497
      CliqueNodeIt(const GrossoLocatelliPullanMc &mc)
kpeter@999
   498
       : _map(mc._graph)
kpeter@999
   499
      {
kpeter@999
   500
        mc.cliqueMap(_map);
kpeter@999
   501
        for (_it = NodeIt(mc._graph); _it != INVALID && !_map[_it]; ++_it) ;
kpeter@999
   502
      }
kpeter@999
   503
kpeter@999
   504
      /// Conversion to \c Node
kpeter@999
   505
      operator Node() const { return _it; }
kpeter@999
   506
kpeter@999
   507
      bool operator==(Invalid) const { return _it == INVALID; }
kpeter@999
   508
      bool operator!=(Invalid) const { return _it != INVALID; }
kpeter@999
   509
kpeter@999
   510
      /// Next node
kpeter@999
   511
      CliqueNodeIt &operator++() {
kpeter@999
   512
        for (++_it; _it != INVALID && !_map[_it]; ++_it) ;
kpeter@999
   513
        return *this;
kpeter@999
   514
      }
kpeter@999
   515
kpeter@999
   516
      /// Postfix incrementation
kpeter@999
   517
kpeter@999
   518
      /// Postfix incrementation.
kpeter@999
   519
      ///
kpeter@999
   520
      /// \warning This incrementation returns a \c Node, not a
kpeter@999
   521
      /// \c CliqueNodeIt as one may expect.
kpeter@999
   522
      typename GR::Node operator++(int) {
kpeter@999
   523
        Node n=*this;
kpeter@999
   524
        ++(*this);
kpeter@999
   525
        return n;
kpeter@999
   526
      }
kpeter@999
   527
kpeter@999
   528
    };
kpeter@999
   529
kpeter@999
   530
    /// @}
kpeter@999
   531
kpeter@999
   532
  private:
kpeter@999
   533
kpeter@999
   534
    // Adds a node to the current clique
kpeter@999
   535
    void addCliqueNode(int u) {
kpeter@999
   536
      if (_clique[u]) return;
kpeter@999
   537
      _clique[u] = true;
kpeter@999
   538
      _size++;
kpeter@999
   539
      BoolVector &row = _gr[u];
kpeter@999
   540
      for (int i = 0; i != _n; i++) {
kpeter@999
   541
        if (!row[i]) _delta[i]++;
kpeter@999
   542
      }
kpeter@999
   543
    }
kpeter@999
   544
kpeter@999
   545
    // Removes a node from the current clique
kpeter@999
   546
    void delCliqueNode(int u) {
kpeter@999
   547
      if (!_clique[u]) return;
kpeter@999
   548
      _clique[u] = false;
kpeter@999
   549
      _size--;
kpeter@999
   550
      BoolVector &row = _gr[u];
kpeter@999
   551
      for (int i = 0; i != _n; i++) {
kpeter@999
   552
        if (!row[i]) _delta[i]--;
kpeter@999
   553
      }
kpeter@999
   554
    }
kpeter@999
   555
kpeter@999
   556
    // Initialize data structures
kpeter@999
   557
    void init() {
kpeter@999
   558
      _n = countNodes(_graph);
kpeter@999
   559
      int ui = 0;
kpeter@999
   560
      for (NodeIt u(_graph); u != INVALID; ++u) {
kpeter@999
   561
        _id[u] = ui++;
kpeter@999
   562
      }
kpeter@999
   563
      _gr.clear();
kpeter@999
   564
      _gr.resize(_n, BoolVector(_n, false));
kpeter@999
   565
      ui = 0;
kpeter@999
   566
      for (NodeIt u(_graph); u != INVALID; ++u) {
kpeter@999
   567
        for (IncEdgeIt e(_graph, u); e != INVALID; ++e) {
kpeter@999
   568
          int vi = _id[_graph.runningNode(e)];
kpeter@999
   569
          _gr[ui][vi] = true;
kpeter@999
   570
          _gr[vi][ui] = true;
kpeter@999
   571
        }
kpeter@999
   572
        ++ui;
kpeter@999
   573
      }
kpeter@999
   574
kpeter@999
   575
      _clique.clear();
kpeter@999
   576
      _clique.resize(_n, false);
kpeter@999
   577
      _size = 0;
kpeter@999
   578
      _best_clique.clear();
kpeter@999
   579
      _best_clique.resize(_n, false);
kpeter@999
   580
      _best_size = 0;
kpeter@999
   581
      _delta.clear();
kpeter@999
   582
      _delta.resize(_n, 0);
kpeter@999
   583
      _tabu.clear();
kpeter@999
   584
      _tabu.resize(_n, false);
kpeter@999
   585
    }
kpeter@999
   586
kpeter@999
   587
    // Executes the algorithm
kpeter@999
   588
    template <typename SelectionRuleImpl>
kpeter@999
   589
    int start(int max_select) {
kpeter@999
   590
      // Options for the restart rule
kpeter@999
   591
      const bool delta_based_restart = true;
kpeter@999
   592
      const int restart_delta_limit = 4;
kpeter@999
   593
kpeter@999
   594
      if (_n == 0) return 0;
kpeter@999
   595
      if (_n == 1) {
kpeter@999
   596
        _best_clique[0] = true;
kpeter@999
   597
        _best_size = 1;
kpeter@999
   598
        return _best_size;
kpeter@999
   599
      }
kpeter@999
   600
kpeter@999
   601
      // Iterated local search
kpeter@999
   602
      SelectionRuleImpl sel_method(*this);
kpeter@999
   603
      int select = 0;
kpeter@999
   604
      IntVector restart_nodes;
kpeter@999
   605
kpeter@999
   606
      while (select < max_select) {
kpeter@999
   607
kpeter@999
   608
        // Perturbation/restart
kpeter@999
   609
        if (delta_based_restart) {
kpeter@999
   610
          restart_nodes.clear();
kpeter@999
   611
          for (int i = 0; i != _n; i++) {
kpeter@999
   612
            if (_delta[i] >= restart_delta_limit)
kpeter@999
   613
              restart_nodes.push_back(i);
kpeter@999
   614
          }
kpeter@999
   615
        }
kpeter@999
   616
        int rs_node = -1;
kpeter@999
   617
        if (restart_nodes.size() > 0) {
kpeter@999
   618
          rs_node = restart_nodes[_rnd[restart_nodes.size()]];
kpeter@999
   619
        } else {
kpeter@999
   620
          rs_node = _rnd[_n];
kpeter@999
   621
        }
kpeter@999
   622
        BoolVector &row = _gr[rs_node];
kpeter@999
   623
        for (int i = 0; i != _n; i++) {
kpeter@999
   624
          if (_clique[i] && !row[i]) delCliqueNode(i);
kpeter@999
   625
        }
kpeter@999
   626
        addCliqueNode(rs_node);
kpeter@999
   627
kpeter@999
   628
        // Local search
kpeter@999
   629
        _tabu.clear();
kpeter@999
   630
        _tabu.resize(_n, false);
kpeter@999
   631
        bool tabu_empty = true;
kpeter@999
   632
        int max_swap = _size;
kpeter@999
   633
        while (select < max_select) {
kpeter@999
   634
          select++;
kpeter@999
   635
          int u;
kpeter@999
   636
          if ((u = sel_method.nextFeasibleAddNode()) != -1) {
kpeter@999
   637
            // Feasible add move
kpeter@999
   638
            addCliqueNode(u);
kpeter@999
   639
            if (tabu_empty) max_swap = _size;
kpeter@999
   640
          }
kpeter@999
   641
          else if ((u = sel_method.nextFeasibleSwapNode()) != -1) {
kpeter@999
   642
            // Feasible swap move
kpeter@999
   643
            int v = -1;
kpeter@999
   644
            BoolVector &row = _gr[u];
kpeter@999
   645
            for (int i = 0; i != _n; i++) {
kpeter@999
   646
              if (_clique[i] && !row[i]) {
kpeter@999
   647
                v = i;
kpeter@999
   648
                break;
kpeter@999
   649
              }
kpeter@999
   650
            }
kpeter@999
   651
            addCliqueNode(u);
kpeter@999
   652
            delCliqueNode(v);
kpeter@999
   653
            _tabu[v] = true;
kpeter@999
   654
            tabu_empty = false;
kpeter@999
   655
            if (--max_swap <= 0) break;
kpeter@999
   656
          }
kpeter@999
   657
          else if ((u = sel_method.nextAddNode()) != -1) {
kpeter@999
   658
            // Non-feasible add move
kpeter@999
   659
            addCliqueNode(u);
kpeter@999
   660
          }
kpeter@999
   661
          else break;
kpeter@999
   662
        }
kpeter@999
   663
        if (_size > _best_size) {
kpeter@999
   664
          _best_clique = _clique;
kpeter@999
   665
          _best_size = _size;
kpeter@999
   666
          if (_best_size == _n) return _best_size;
kpeter@999
   667
        }
kpeter@999
   668
        sel_method.update();
kpeter@999
   669
      }
kpeter@999
   670
kpeter@999
   671
      return _best_size;
kpeter@999
   672
    }
kpeter@999
   673
kpeter@999
   674
  }; //class GrossoLocatelliPullanMc
kpeter@999
   675
kpeter@999
   676
  ///@}
kpeter@999
   677
kpeter@999
   678
} //namespace lemon
kpeter@999
   679
kpeter@999
   680
#endif //LEMON_GROSSO_LOCATELLI_PULLAN_MC_H