src/work/akos/simann.h
author klao
Mon, 06 Dec 2004 00:30:44 +0000
changeset 1030 c8a41699e613
parent 1018 68beae6758a7
child 1096 1cfb25ef14d2
permissions -rw-r--r--
Undirected graph documentation and concept refinements.

* quite a few bug fixes
* concept::UndirGraph is almost complete and looks quite good.
ladanyi@942
     1
#ifndef LEMON_SIMANN_H
ladanyi@942
     2
#define LEMON_SIMANN_H
ladanyi@918
     3
ladanyi@966
     4
#include <cstdlib>
ladanyi@966
     5
#include <cmath>
ladanyi@1018
     6
#include <lemon/time_measure.h>
ladanyi@966
     7
ladanyi@942
     8
namespace lemon {
ladanyi@918
     9
ladanyi@942
    10
  const double INFTY = 1e24;
ladanyi@918
    11
ladanyi@942
    12
  class SimAnnBase {
ladanyi@918
    13
  public:
ladanyi@942
    14
    class Controller;
ladanyi@942
    15
  private:
ladanyi@942
    16
    Controller *controller;
ladanyi@942
    17
  protected:
ladanyi@942
    18
    double curr_cost;
ladanyi@1023
    19
    double best_cost;
ladanyi@942
    20
    double prev_cost;
ladanyi@1023
    21
    double prev_prev_cost;
ladanyi@918
    22
ladanyi@942
    23
    virtual void mutate() = 0;
ladanyi@942
    24
    virtual void revert() = 0;
ladanyi@942
    25
    virtual void saveAsBest() = 0;
ladanyi@942
    26
  public:
ladanyi@942
    27
    SimAnnBase() {
ladanyi@1023
    28
      best_cost = prev_cost = prev_prev_cost = INFTY;
ladanyi@942
    29
    }
ladanyi@957
    30
    void setController(Controller &_controller) {
ladanyi@957
    31
      controller = &_controller;
ladanyi@957
    32
      controller->setBase(this);
ladanyi@957
    33
    }
ladanyi@1018
    34
    double getCurrCost() const { return curr_cost; }
ladanyi@1018
    35
    double getPrevCost() const { return prev_cost; }
ladanyi@1018
    36
    double getBestCost() const { return best_cost; }
ladanyi@942
    37
    void run() {
ladanyi@966
    38
      controller->init();
ladanyi@1018
    39
      do {
ladanyi@942
    40
        mutate();
ladanyi@957
    41
        if (controller->accept()) {
ladanyi@942
    42
          controller->acceptEvent();
ladanyi@942
    43
          if (curr_cost < best_cost) {
ladanyi@942
    44
            saveAsBest();
ladanyi@942
    45
            controller->improveEvent();
ladanyi@942
    46
          }
ladanyi@942
    47
        }
ladanyi@942
    48
        else {
ladanyi@942
    49
          revert();
ladanyi@942
    50
          controller->rejectEvent();
ladanyi@942
    51
        }
ladanyi@1018
    52
      } while (controller->next());
ladanyi@918
    53
    }
ladanyi@918
    54
ladanyi@1000
    55
    /*! \brief A base class for controllers. */
ladanyi@942
    56
    class Controller {
ladanyi@942
    57
    public:
ladanyi@957
    58
      SimAnnBase *base;
ladanyi@966
    59
      virtual void init() {}
ladanyi@1000
    60
      /*! \brief This is called when a neighbouring state gets accepted. */
ladanyi@942
    61
      virtual void acceptEvent() {}
ladanyi@1000
    62
      /*! \brief This is called when the accepted neighbouring state's cost is
ladanyi@1000
    63
       *  less than the best found one's.
ladanyi@1000
    64
       */
ladanyi@942
    65
      virtual void improveEvent() {}
ladanyi@1000
    66
      /*! \brief This is called when a neighbouring state gets rejected. */
ladanyi@942
    67
      virtual void rejectEvent() {}
ladanyi@957
    68
      virtual void setBase(SimAnnBase *_base) { base = _base; }
ladanyi@1000
    69
      /*! */
ladanyi@942
    70
      virtual bool next() = 0;
ladanyi@1000
    71
      /*! */
ladanyi@957
    72
      virtual bool accept() = 0;
ladanyi@942
    73
    };
ladanyi@942
    74
  };
ladanyi@918
    75
ladanyi@942
    76
  template <typename E>
ladanyi@942
    77
  class SimAnn : public SimAnnBase {
ladanyi@942
    78
  private:
ladanyi@942
    79
    E *curr_ent;
ladanyi@942
    80
    E *best_ent;
ladanyi@942
    81
  public:
ladanyi@957
    82
    SimAnn() : SimAnnBase() {}
ladanyi@957
    83
    void setEntity(E &ent) {
ladanyi@957
    84
      curr_ent = new E(ent);
ladanyi@957
    85
      best_ent = new E(ent);
ladanyi@1023
    86
      curr_cost = curr_ent->getCost();
ladanyi@942
    87
    }
ladanyi@942
    88
    E getBestEntity() { return *best_ent; }
ladanyi@942
    89
    void mutate() {
ladanyi@1023
    90
      prev_prev_cost = prev_cost;
ladanyi@1018
    91
      prev_cost = curr_cost;
ladanyi@1023
    92
      curr_ent->mutate();
ladanyi@1023
    93
      curr_cost = curr_ent->getCost();
ladanyi@942
    94
    }
ladanyi@942
    95
    void revert() {
ladanyi@942
    96
      curr_ent->revert();
ladanyi@1018
    97
      curr_cost = prev_cost;
ladanyi@1023
    98
      prev_cost = prev_prev_cost;
ladanyi@942
    99
    }
ladanyi@942
   100
    void saveAsBest() {
ladanyi@942
   101
      *best_ent = *curr_ent;
ladanyi@942
   102
      best_cost = curr_cost;
ladanyi@942
   103
    }
ladanyi@942
   104
  };
ladanyi@942
   105
ladanyi@956
   106
  class EntitySkeleton {
ladanyi@942
   107
  public:
ladanyi@1023
   108
    /*! \return the cost of the entity */
ladanyi@1023
   109
    double getCost() { return 0.0; }
ladanyi@1023
   110
    /*! \brief Makes a minor change to the entity. */
ladanyi@1023
   111
    void mutate() {}
ladanyi@966
   112
    /*! \brief Restores the entity to its previous state i.e. reverts the
ladanyi@966
   113
     *  effects of the last mutate.
ladanyi@966
   114
     */
ladanyi@942
   115
    void revert() {}
ladanyi@942
   116
  };
ladanyi@942
   117
ladanyi@966
   118
  /*! \brief A simple controller for the simulated annealing class.
ladanyi@966
   119
   *  \todo Find a way to set the various parameters.
ladanyi@966
   120
   */
ladanyi@956
   121
  class SimpleController : public SimAnnBase::Controller {
ladanyi@956
   122
  public:
ladanyi@956
   123
    long iter, last_impr, max_iter, max_no_impr;
ladanyi@1000
   124
    double temp, ann_fact;
ladanyi@1000
   125
    /*! \param _max_iter maximum number of iterations
ladanyi@1000
   126
     *  \param _max_no_impr maximum number of consecutive iterations which do
ladanyi@1000
   127
     *         not yield a better solution
ladanyi@1000
   128
     *  \param _temp initial temperature
ladanyi@1000
   129
     *  \param _ann_fact annealing factor
ladanyi@1000
   130
     */
ladanyi@1000
   131
    SimpleController(long _max_iter = 500000, long _max_no_impr = 20000,
ladanyi@1000
   132
    double _temp = 1000, double _ann_fact = 0.9999) : iter(0), last_impr(0),
ladanyi@1000
   133
    max_iter(_max_iter), max_no_impr(_max_no_impr), temp(_temp),
ladanyi@1000
   134
    ann_fact(_ann_fact) {}
ladanyi@956
   135
    void acceptEvent() {
ladanyi@956
   136
      iter++;
ladanyi@956
   137
    }
ladanyi@956
   138
    void improveEvent() {
ladanyi@956
   139
      last_impr = iter;
ladanyi@956
   140
    }
ladanyi@956
   141
    void rejectEvent() {
ladanyi@956
   142
      iter++;
ladanyi@956
   143
    }
ladanyi@956
   144
    bool next() {
ladanyi@1000
   145
      temp *= ann_fact;
ladanyi@956
   146
      bool quit = (iter > max_iter) || (iter - last_impr > max_no_impr);
ladanyi@956
   147
      return !quit;
ladanyi@956
   148
    }
ladanyi@957
   149
    bool accept() {
ladanyi@1018
   150
      double cost_diff = base->getPrevCost() - base->getCurrCost();
ladanyi@1018
   151
      if (cost_diff < 0.0) {
ladanyi@1018
   152
        return (drand48() <= exp(cost_diff / temp));
ladanyi@1018
   153
      }
ladanyi@1018
   154
      else {
ladanyi@1018
   155
        return true;
ladanyi@1018
   156
      }
ladanyi@966
   157
    }
ladanyi@966
   158
  };
ladanyi@966
   159
ladanyi@966
   160
  /*! \brief A controller with preset running time for the simulated annealing
ladanyi@966
   161
   *  class.
ladanyi@966
   162
   *  \todo Find a better name.
ladanyi@966
   163
   */
ladanyi@966
   164
  class AdvancedController : public SimAnnBase::Controller {
ladanyi@966
   165
  private:
ladanyi@1018
   166
    Timer timer;
ladanyi@1000
   167
    /*! \param time the elapsed time in seconds */
ladanyi@1018
   168
    virtual double threshold(double time) {
ladanyi@1018
   169
      // this is the function 1 / log(x) scaled and offset
ladanyi@1018
   170
      static double xm = 5.0 / end_time;
ladanyi@1018
   171
      static double ym = start_threshold / (1 / log(1.2) - 1 / log(5.0 + 1.2));
ladanyi@1018
   172
      return ym * (1 / log(xm * time + 1.2) - 1 / log(5.0 + 1.2));
ladanyi@1018
   173
    }
ladanyi@966
   174
  public:
ladanyi@1000
   175
    double alpha, beta, gamma;
ladanyi@1000
   176
    double end_time, start_time;
ladanyi@1018
   177
    double start_threshold;
ladanyi@966
   178
    double avg_cost;
ladanyi@1000
   179
    double temp, ann_fact;
ladanyi@1018
   180
    bool warmup;
ladanyi@1018
   181
    /*! \param _end_time running time in seconds
ladanyi@1000
   182
     *  \param _alpha parameter used to calculate the running average
ladanyi@1000
   183
     *  \param _beta parameter used to decrease the annealing factor
ladanyi@1000
   184
     *  \param _gamma parameter used to increase the temperature
ladanyi@1000
   185
     */
ladanyi@1000
   186
    AdvancedController(double _end_time, double _alpha = 0.2,
ladanyi@1000
   187
    double _beta = 0.9, double _gamma = 1.2) : alpha(_alpha), beta(_beta),
ladanyi@1023
   188
    gamma(_gamma), end_time(_end_time), ann_fact(0.9999), warmup(true) {}
ladanyi@966
   189
    void init() {
ladanyi@1018
   190
      avg_cost = base->getCurrCost();
ladanyi@966
   191
    }
ladanyi@966
   192
    void acceptEvent() {
ladanyi@966
   193
      avg_cost = alpha * base->getCurrCost() + (1.0 - alpha) * avg_cost;
ladanyi@1023
   194
      if (warmup) {
ladanyi@1023
   195
        static double max_cost_diff = 0.0;
ladanyi@1023
   196
        static int incr_cnt = 0;
ladanyi@1023
   197
        double cost_diff = base->getCurrCost() - base->getPrevCost();
ladanyi@1023
   198
        if (cost_diff > 0.0) {
ladanyi@1023
   199
          incr_cnt++;
ladanyi@1023
   200
          if (cost_diff > max_cost_diff) {
ladanyi@1023
   201
            max_cost_diff = cost_diff;
ladanyi@1023
   202
          }
ladanyi@1023
   203
        }
ladanyi@1023
   204
        if (incr_cnt >= 100) {
ladanyi@1023
   205
          // calculate starting threshold and starting temperature
ladanyi@1023
   206
          start_threshold = fabs(base->getBestCost() - avg_cost);
ladanyi@1023
   207
          temp = max_cost_diff / log(0.5);
ladanyi@1023
   208
          warmup = false;
ladanyi@1023
   209
          timer.reset();
ladanyi@1023
   210
        }
ladanyi@1023
   211
      }
ladanyi@966
   212
    }
ladanyi@966
   213
    void improveEvent() {
ladanyi@966
   214
    }
ladanyi@966
   215
    void rejectEvent() {
ladanyi@966
   216
    }
ladanyi@966
   217
    bool next() {
ladanyi@1018
   218
      if (warmup) {
ladanyi@1018
   219
        return true;
ladanyi@1000
   220
      }
ladanyi@1000
   221
      else {
ladanyi@1018
   222
        double elapsed_time = timer.getRealTime();
ladanyi@1018
   223
        if (fabs(avg_cost - base->getBestCost()) > threshold(elapsed_time)) {
ladanyi@1018
   224
          // decrease the annealing factor
ladanyi@1018
   225
          ann_fact *= beta;
ladanyi@1018
   226
        }
ladanyi@1018
   227
        else {
ladanyi@1018
   228
          // increase the temperature
ladanyi@1018
   229
          temp *= gamma;
ladanyi@1018
   230
        }
ladanyi@1018
   231
        temp *= ann_fact;
ladanyi@1018
   232
        return elapsed_time < end_time;
ladanyi@1000
   233
      }
ladanyi@966
   234
    }
ladanyi@966
   235
    bool accept() {
ladanyi@1018
   236
      if (warmup) {
ladanyi@1018
   237
        // we accept eveything during the "warm up" phase
ladanyi@1018
   238
        return true;
ladanyi@1018
   239
      }
ladanyi@1018
   240
      else {
ladanyi@1018
   241
        double cost_diff = base->getPrevCost() - base->getCurrCost();
ladanyi@1018
   242
        if (cost_diff < 0.0) {
ladanyi@1018
   243
          return (drand48() <= exp(cost_diff / temp));
ladanyi@1018
   244
        }
ladanyi@1018
   245
        else {
ladanyi@1018
   246
          return true;
ladanyi@1018
   247
        }
ladanyi@1018
   248
      }
ladanyi@956
   249
    }
ladanyi@956
   250
  };
ladanyi@956
   251
ladanyi@942
   252
}
ladanyi@918
   253
ladanyi@918
   254
#endif