模拟退火和遗传算法解决TSP问题

模拟退火和遗传算法解决TSP问题

数据集介绍

  • 采用数据集FRI26
    来自标准数据集,共有26个城市,最优解为933:
    数据下载链接
    模拟退火和遗传算法解决TSP问题_第1张图片
图1:数据矩阵

模拟退火和遗传算法解决TSP问题_第2张图片

图2:数据集介绍

算法介绍

模拟退火

介绍:
模拟退火是一种通用概率算法,常用来在一定时间内寻找在一个很大搜寻空间中的近似最优解。

迭代过程:
迭代过程是模拟退火算法的核心步骤,分为新解的产生和接受新解两部分:

  1. 由一个产生函数从当前解产生一个位于解空间的新解;为便于后续的计算和接受,减少算法耗时,通常选择由当前新解经过简单地变换即可产生新解的方法,如对构成新解的全部或部分元素进行置换、互换等,注意到产生新解的变换方法决定了当前新解的邻域结构,因而对冷却进度表的选取有一定的影响。
  2. 计算与新解所对应的目标函数差。因为目标函数差仅由变换部分产生,所以目标函数差的计算最好按增量计算。事实表明,对大多数应用而言,这是计算目标函数差的最快方法。
  3. 判断新解是否被接受,判断的依据是一个接受准则,最常用的接受准则是Metropolis准则:若Δt′<0则接受S′作为新的当前解S,否则以概率exp(-Δt′/T)接受S′作为新的当前解S。
  4. 当新解被确定接受时,用新解代替当前解,这只需将当前解中对应于产生新解时的变换部分予以实现,同时修正目标函数值即可。此时,当前解实现了一次迭代。可在此基础上开始下一轮试验。而当新解被判定为舍弃时,则在原当前解的基础上继续下一轮试验。

实现代码:

// tsp_sa.cpp: 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;

#define T0 50000.0		//初始化温度
#define Tk (1e-8)		//终止温度
#define d  0.98			// 退火系数
#define L 1000			// 每个温度时的迭代次数,即链长
#define N 26			// 城市数量
#define TESTTIME  30    //测试次数

int solution[N];		// 用于存放一个解

int map[N][N];			//记录地图数据				

const char* filepath = "C://Users//56989//Desktop//dataset.txt";
ifstream infile;

//读取数据
void readData()
{
	infile.open(filepath);
	assert(infile.is_open()); //若失败,则输出错误消息,并终止程序运行 
	for (int i = 0; i < 26; i++)
	{
		for (int j = 0; j < 26; j++)
		{
			infile >> map[i][j];
		}
	}
}

// 计算路径长度
int pathLen(int * arr)
{
	int totlen = 0;
	for (int i = 0; i < N-1; i++)
	{
		totlen += map[arr[i]][arr[i + 1]];
	}
	totlen += map[arr[N-1]][arr[0]];	//回到出发城市
	return totlen;
}

// 初始化函数
void initSolution()
{
	for (int i = 0; i Tk)
		{
			//printf("进入.\n");
			for (int i = 1; i <= L; i++)
			{
				memcpy(tempsolu, solution, N * sizeof(int)); // 复制数组,保留原来的解
				genAnotherSolu();							 // 产生新解

				tempLen = pathLen(tempsolu);
				soluLen = pathLen(solution);

				int dif = soluLen - tempLen;

				if (dif >= 0)//原来的解更好
				{
					double ran = ((double)rand()) / (RAND_MAX);
					if (exp(-dif / T) <= ran)   // 保留原来的解
					{
						memcpy(solution, tempsolu, N * sizeof(int));
					}
				}
			}
			T = T * d; // 降温

		}
		aveLen += pathLen(solution);
		printf("第%d次计算完成,所得路径长度为: %d\n", testtime, pathLen(solution));

	}
	aveLen = aveLen / 30;

	finish = clock(); 

	double duration = ((double)(finish - start)) / CLOCKS_PER_SEC;	// 计算时间
	
	printf("程序运行耗时:%lf秒.\n", duration);
	printf("重复30次,求得的最优路径平均值为:%2f.\n", aveLen);

	/*
	printf("模拟退火算法\n");
	printf("路径如下:\n");
	for (int i = 0; i

实验结果:
计算30次,取平均值:
模拟退火和遗传算法解决TSP问题_第3张图片

图3:模拟退火结果

遗传算法

介绍:
遗传算法是计算数学中用于解决最优化的搜索算法,是进化算法的一种。进化算法最初是借鉴了进化生物学中的一些现象而发展起来的,这些现象包括遗传、突变、自然选择以及杂交等。

实现代码:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;

#define MAXGENS 500  // 最大进化代数
#define POPSIZE 100  // 种群数目
#define PXOVER 0.6   // 交叉概率
#define PMUTATION 0.1 // 变异概率
#define N 26 // 染色体长度(这里即为城市个数)
#define TESTTIME  30    //测试次数

int pop[POPSIZE][N];	 // 种群
int fitness[POPSIZE];

int globalBest[N];		// 最佳路线
int bestFitness = 0x7FFFFFFF; // 最短路径长度

int map[N][N];			//记录地图数据	

const char* filepath = "C://Users//56989//Desktop//dataset.txt";
ifstream infile;

//读取数据
void readData()
{
	infile.open(filepath);
	assert(infile.is_open()); //若失败,则输出错误消息,并终止程序运行 
	for (int i = 0; i < N; i++)
	{
		for (int j = 0; j < N; j++)
		{
			infile >> map[i][j];
		}
	}
	infile.close();
}

// 种群初始化
void init()
{
	int num = 0;
	while (num < POPSIZE)
	{
		for (int i = 0; i= POPSIZE)
					break;
			}
			if (num >= POPSIZE)
				break;
		}

		while (num < POPSIZE)
		{
			double r1 = ((double)rand()) / (RAND_MAX + 1.0);
			double r2 = ((double)rand()) / (RAND_MAX + 1.0);
			int p1 = (int)(N*r1); // 位置1
			int p2 = (int)(N*r2); // 位置2
			int temp = pop[num][p1];
			pop[num][p1] = pop[num][p2];
			pop[num][p2] = temp;    // 交换基因位置
			num++;
		}
	}

	//	for (int i = 0; i PXOVER)
		{
			move += 2;
			continue; // 本次不进行交叉
		}
		// 采用部分映射杂交
		choice1 = move; // 用于选取杂交的两个父代
		choice2 = move + 1; // 注意避免下标越界
		pick1 = ((double)rand()) / (RAND_MAX + 1.0);
		pick2 = ((double)rand()) / (RAND_MAX + 1.0);
		pos1 = (int)(pick1*N); // 用于确定两个杂交点的位置
		pos2 = (int)(pick2*N);
		while (pos1 > N - 2 || pos1 < 1)
		{
			pick1 = ((double)rand()) / (RAND_MAX + 1.0);
			pos1 = (int)(pick1*N);
		}
		while (pos2 > N - 2 || pos2 < 1)
		{
			pick2 = ((double)rand()) / (RAND_MAX + 1.0);
			pos2 = (int)(pick2*N);
		}
		if (pos1 > pos2)
		{
			temp = pos1;
			pos1 = pos2;
			pos2 = temp; // 交换pos1和pos2的位置
		}
		for (int j = pos1; j <= pos2; j++)
		{
			temp = pop[choice1][j];
			pop[choice1][j] = pop[choice2][j];
			pop[choice2][j] = temp; // 逐个交换顺序
		}
		num1 = 0;
		num2 = 0;
		if (pos1 > 0 && pos2 < N - 1)
		{
			for (int j = 0; j <= pos1 - 1; j++)
			{
				for (int k = pos1; k <= pos2; k++)
				{
					if (pop[choice1][j] == pop[choice1][k])
					{
						conflict1[num1] = j;
						num1++;
					}
					if (pop[choice2][j] == pop[choice2][k])
					{
						conflict2[num2] = j;
						num2++;
					}
				}
			}
			for (int j = pos2 + 1; j 0)
		{
			for (int j = 0; j PMUTATION)
			continue;
		pick1 = ((double)rand()) / (RAND_MAX + 1.0);
		pick2 = ((double)rand()) / (RAND_MAX + 1.0);
		pos1 = (int)(pick1*N); // 选取进行变异的位置
		pos2 = (int)(pick2*N);
		while (pos1 > N - 1)
		{
			pick1 = ((double)rand()) / (RAND_MAX + 1.0);
			pos1 = (int)(pick1*N);
		}
		while (pos2 > N - 1)
		{
			pick2 = ((double)rand()) / (RAND_MAX + 1.0);
			pos2 = (int)(pick2*N);
		}
		temp = pop[i][pos1];
		pop[i][pos1] = pop[i][pos2];
		pop[i][pos2] = temp;
	}
}

//精英更新
void elitist()
{
	int best, worst;
	int bestIndex, worstIndex;

	best = fitness[0];
	worst = fitness[0];

	bestIndex = 0;
	worstIndex = 0;

	//找出最好和最坏
	for (int i = 0; i < POPSIZE; i++)
	{
		if (fitness[i] < best)
		{
			best = fitness[i];
			bestIndex = i;
		}
		else if (fitness[i] > worst)
		{
			worst = fitness[i];
			worstIndex = i;
		}
	}

	//要不更新最好
	if (best < bestFitness)
	{
		for (int i = 0; i < N; i++)
		{
			globalBest[i] = pop[bestIndex][i];
		}
		bestFitness = best;
	}
	else//要不更新最坏
	{
		for (int i = 0; i < N; i++)
		{
			pop[worstIndex][i] = globalBest[i];
		}
		fitness[worstIndex] = bestFitness;
	}
}


int main(void)
{
	
	readData();
	time_t start, finish;
	start = clock();
	double aveLen = 0.0;
	int timeTime = 0;
	while (timeTime < TESTTIME)
	{
		srand((unsigned)time(NULL));	// 初始化随机数种子
		init();

		//更新fitness
		for (int j = 0; j

遗传算法结果:
模拟退火和遗传算法解决TSP问题_第4张图片

图4:遗传算法结果图

走过的坑:

  1. 遗传算法收敛特别慢。我所实现的版本经过了多次!多次!的改良。才勉强达到模拟退火的水平。不然效果特别差。
  2. 遗传算法的初始化对最终结果有较大的影响,这点和模拟退火有很大的不同。
  3. 遗传算法实现难度大于模拟退火。整个过程耗费了本人很多很多时间!

两者对比:

  1. 两者结果其实类似。无论从时间上还是从结果上。
  2. 其实遗传算法要次于模拟退火。一是遗传算法很依赖初始解,这给构造带来了极大的挑战。二是遗传算法收敛特别慢,容易陷入局部最优。
  3. 遗传算法的交叉算子,选择算子有多种并且实现相对复杂,并且需要调的参数比模拟退火多。针对简单问题时,不建议使用遗传算法。

代码下载:

包括sa的源码、GA源码以及数据集

有不足之处请多指教 ?

数据集来源:
http://people.sc.fsu.edu/~jburkardt/datasets/tsp/tsp.html
http://people.sc.fsu.edu/~jburkardt/datasets/tsp/fri26_d.txt

模拟退火博客:
https://www.cnblogs.com/lyrichu/p/6688459.html
https://www.cnblogs.com/flashhu/p/8884132.html
https://www.cnblogs.com/rvalue/p/8678318.html

TSP介绍:
https://baike.baidu.com/item/TSP问题/840008?fr=aladdin

GA:
https://www.cnblogs.com/lyrichu/p/6152928.html
https://blog.csdn.net/greedystar/article/details/80343841#五、源代码

你可能感兴趣的:(智能算法)