最近的一个项目中需要实现个路径规划的算法,需要求得的路径的总长度尽可能的短。这就是典型的旅行商(TSP)问题了。解决这个问题的一个比较好用的方法就是模拟退火算法。网上关于用模拟退火算法解决 TSP 问题的文章挺多的,其中也有不少号称给出了 C++ 代码。但是说句实话,这些代码中没有一个是按照面向对象的思想来实现了,并没有把模拟退火算法的框架封装好。因此,遇到一个新问题时,改写算法就很麻烦。
看了几个网上的实现后,发现 gsl 库给出的框架还是不错的。但是 gsl 是 C 语言的库,虽然是按照面向对象的思想来设计的,但是实现的接口真是挺繁琐的,用起来并不好用。因此,我就花了半天时间,把 gsl 里的相关代码扒了出来,重新封装到了一个类中。
模拟退火算法最早的思想是由 Metropolis 等人提出的。1983 年, Kirkpatrick 成功地将退火思想引入到组合优化领域。简单的说,模拟退火算法就是一种基于 Monte-Carlo 迭代求解策略的随机寻优算法。关于模拟退火算法的理论网上有很多文章介绍。这里就不多写了,今天主要说说如何C++ 实现。
模拟退火算法的核心功能封装到了 SimulatedAnnualingSolver 类中。
#ifndef SIMULATEDANNUALINGSOLVER_H
#define SIMULATEDANNUALINGSOLVER_H
#include
class SimulatedAnnualingSolver
{
public:
SimulatedAnnualingSolver(int rand_seed);
void setParameters(double t_initial, double mu_t, double k = 1, double t_min = 1e-5, int iters_fixed_T = 100, double step_size = 1);
void enablePrint(bool on) {m_print_position = on;}
void solve();
protected:
virtual double take_step(double step_size) = 0;
virtual void undo_step() = 0;
virtual void save_best() = 0;
virtual double energy() = 0;
virtual void print() = 0;
private:
bool m_print_position;
int m_iters_fixed_T;
double m_step_size;
double m_k;
double m_t_initial;
double m_mu_t;
double m_t_min;
std::mt19937 m_randGenerator;
double boltzmann(double E, double new_E, double T, double k);
};
#endif // SIMULATEDANNUALINGSOLVER_H
#include "SimulatedAnnualingSolver.h"
#include
#include
#define GSL_LOG_DBL_MIN (-7.0839641853226408e+02)
SimulatedAnnualingSolver::SimulatedAnnualingSolver(int rand_seed)
:m_t_initial(1),
m_mu_t(1.01),
m_k(1),
m_t_min(0.01),
m_iters_fixed_T(100),
m_print_position(false),
m_step_size(1),
m_randGenerator(rand_seed)
{
}
void SimulatedAnnualingSolver::setParameters(double t_initial, double mu_t, double k, double t_min, int iters_fixed_T, double step_size)
{
m_t_initial = t_initial;
m_mu_t = mu_t;
m_k = k;
m_t_min = t_min;
m_iters_fixed_T = iters_fixed_T;
m_step_size = step_size;
}
void SimulatedAnnualingSolver::solve()
{
int n_evals = 1, n_iter = 0;
double E = energy();
double best_E = E;
save_best(); // 将当前状态存为最佳解
double T = m_t_initial;
double T_factor = 1.0 / m_mu_t;
std::uniform_real_distribution<> dis(0, 1);
while (1)
{
int n_accepts = 0;
int n_rejects = 0;
int n_eless = 0;
for (int i = 0; i < m_iters_fixed_T; ++i)
{
double new_E = take_step (m_step_size);
if(new_E <= best_E)
{
best_E = new_E;
save_best();
}
++n_evals;
if (new_E < E)
{
if (new_E < best_E)
{
best_E = new_E;
save_best();
}
/* yay! take a step */
E = new_E;
++n_eless;
}
else if (dis(m_randGenerator) < boltzmann(E, new_E, T, m_k))
{
/* yay! take a step */
E = new_E;
++n_accepts;
}
else
{
undo_step(); // 回退到上一个状态
++n_rejects;
}
}
if (m_print_position)
{
printf ("%5d %7d %12g", n_iter, n_evals, T);
print();
printf (" %12g %12g\n", E, best_E);
}
/* apply the cooling schedule to the temperature */
T *= T_factor;
++n_iter;
if (T < m_t_min)
{
break;
}
}
}
inline double SimulatedAnnualingSolver::boltzmann(double E, double new_E, double T, double k)
{
double x = -(new_E - E) / (k * T);
/* avoid underflow errors for large uphill steps */
return (x < GSL_LOG_DBL_MIN) ? 0.0 : exp(x);
}
这个类有几个纯虚函数。
对于我们一个具体的问题,我们只需要定义一个派生自这个类的新类,在新类中实现这些虚函数。
比如对于我的问题,一个简化了的旅行商问题,我建立了一个 TspSolver 的类。
#ifndef TSPSOLVER_H
#define TSPSOLVER_H
#include "SimulatedAnnualingSolver.h"
#include
#include
#include
#include
class tspSolver: public SimulatedAnnualingSolver
{
public:
tspSolver(int rand_seed, QVector points);
QVector result();
protected:
double take_step(double step_size) override;
void undo_step() override;
void save_best() override;
double energy() override;
void print() override;
private:
std::mt19937 m_randGen;
QVector m_points;
QVector m_best;
int m_N;
int m_n1;
int m_n2;
};
#endif // TSPSOLVER_H
#include "tspsolver.h"
tspSolver::tspSolver(int rand_seed, QVector points)
:SimulatedAnnualingSolver(rand_seed)
{
std::random_device r;
std::seed_seq seed2{r(), r(), r(), r(), r(), r(), r(), r()};
m_randGen.seed(seed2);
m_points = points;
m_N = m_points.size();
}
void tspSolver::save_best()
{
m_best = m_points;
}
QVector tspSolver::result()
{
return m_best;
}
void tspSolver::print()
{
qDebug() << m_points;
}
double tspSolver::take_step(double step_size)
{
std::uniform_int_distribution<> dis(1, m_N - 1);
m_n1 = dis(m_randGen);
do
{
m_n2 = dis(m_randGen);
}
while(m_n2 == m_n1);
// qDebug() << "n1 = " << m_n1 << "n2 = " << m_n2;
std::swap(m_points[m_n1], m_points[m_n2]);
return energy();
}
void tspSolver::undo_step()
{
std::swap(m_points[m_n1], m_points[m_n2]);
}
double tspSolver::energy()
{
double e = 0;
for(int i = 1; i < m_N; i ++)
{
QPoint a1 = m_points.at(i - 1);
QPoint a2 = m_points.at(i);
double x = abs(a1.x() - a2.x());
double y = abs(a1.y() - a2.y());
e += qMax(x, y);
}
return e;
}
大家可以看到,我这个 TSP 问题中的两点间的距离不是普通的直线距离,而是 X Y 方向投影距离的最大值。
take_step 函数实现的也很简答,就是随机选两个点,交换这两个点的坐标。undo_step 就是把这两个点的坐标再交换回去。
下面是一个算例,随便选了 9 个点。规划出的路径看起来还不错。
这个算例选了 32 个点。规划出的路径挺复杂,但是每一步的长度都是 1,所以是最优解。当然这个问题的最优解不止一个。这里得到的解是最优解中较复杂的一个情形。