TSP 路径构造算法

2024-05-24 22:38
文章标签 算法 路径 构造 tsp

本文主要是介绍TSP 路径构造算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

TSP 路径构造算法(tour construction algorithm)

onezeros@yahoo.cn

前言:Tsp是经典的NP问题之一,其现实意义也是不言而喻的。该问题的研究很有些年头了,现在多集中于新型智能算法,如模拟退火,禁忌搜索,蚁群算法,粒子群算法,遗传算法等。这些算法虽很有效,但使用良好的初始解得时候,效果会更优。本文系统介绍并详细讲解了前辈们的关于构建优良TSP路径的启发式(heuristic)算法,可以为前面提到的智能算法提供优秀的初始解。而这些算法本身就是TSP的近似解算法,独立使用这些算法的效果也是很好的。

Tsp的经典解决方案的流程是:

一:使用路径构造算法(tour construction algorithm)构造初始解

二:使用路径优化算法(tour construction algorithm)改善当前解

本文介绍路径构造算法,优化算法将在另外一篇文章中介绍

一般用来构造很好的启发路径,可以作为其他算法(蚁群算法,遗传算法等)的初始解

常见的有以下几种构造算法:

1           最邻近算法(nearest neighbor algorithm

2           贪婪算法(greedy algorithm

3           Clark&Wright 算法

4           插入算法(insertion algorithm

4.1          最邻近插入(nearest insertion

4.2          最小代价插入(cheapest insertion

4.3          任意插入(arbitrary insertion

4.4          最远插入(farthest insertion

4.5          凸包插入(connex hull insertion

5           双生成树算法(double spanning tree algorithm

6           Christofides 算法

下面一一介绍其算法实现过程,这些算法都是很老的算法了,网上比较少见了,相关论文中也都是一笔带过。这是笔者整理所得,有摘录,有重新组织,有翻译,希望对您有益。

1 最邻近算法(nearest neighbor algorithm

时间复杂度:O(n^2)

       Step 1:选任一点为起始节点

       Step 2:找到距离最新加入路径最近的点,将其加入路径

       Step 3:重复Step 2,直至所以点都加入到路径中,连接最后的点和最初的点

2 贪婪算法(greedy algorithm

时间复杂度:O(n^2*log2(n))

Step 1:给所有边按长度排序

Step 2:选择最短的满足条件的边加入到路径中。条件:不会构成小于N条边的环路

Step 3:重复Step 2,直至路径中有N条边。

3 Clark&Wright 算法

时间复杂度:O(n^2)

Step 1:任选一点作为中心结点,索引为 1(假设城市下标从 1 开始)

     Step 2:计算所有的费用 C[i][j]=Distance[i][1]+Distance[1][j],(i,j=2,3,…,N)

Step 3:将费用表C从大到小排序

Step 4:从费用表C的顶端开始向下,一次将可以连接的路径(i,j)连接,直至形成完整路径

4 插入算法(insertion algorithm

都是在已有的子路径上加入新的点形成更大的图,直至包含所有点。不同的是选择点和插入点的策略

4.1 最邻近插入(nearest insertion

时间复杂度:O(n^2)

       Step 1:任选一个点 i,作为子路径(只包含一个结点的子路径)

Step 2:找到距离i最近的点r,形成子路径 i-r-i

Step 3:选择。找到不在子路径中的点r,使它到子路径中的点的最小距离是所有不在子路径中的点到子路径中的点的最小距离中最小的。这个距离是连接这两类点的最小距离

Step 4:插入。在子路径中找到这样一条边(ij),使Distance[i][r]+Distance[r][j]-Distance[i][j]最小。将r插入到ij中间

Step 5:执行Step 3,直到所有点都加入到了路径中

4.2 最小代价插入(cheapest insertion

时间复杂度:O(n^2*lg(n))

Step 1:任选一个点 i,作为子路径(只包含一个结点的子路径)

       Step 2:找到距离i最近的点r,形成子路径 i-r-i

       Step 3:找到在子路径中的点i,j,和不在子路径中的点r,使得Distance[i][r]+Distance[r][j]-Distance[i][j]最小。将r插入到ij中间

Step 4:重复Step 3,直至所有点都加入路径

      

       前两步和最邻近插入是一样的

      

4.3 任意插入(arbitrary insertion

时间复杂度:O(n^2)

Step 1:任选一个点 i,作为子路径(只包含一个结点的子路径)

       Step 2:找到距离i最近的点r,形成子路径 i-r-i

       Step 3:选择任意不在子路径中的点插入到子路径

Step 4:重复Step 3,直至所有点加入到子路径

       前两步和最邻近插入是一样的

4.4 最远插入(farthest insertion

时间复杂度:O(n^2)

Step 1:任选一个点 i,作为子路径(只包含一个结点的子路径)

       Step 2:找到距离i最远的点r,形成子路径 i-r-i

       Step 3:选择。找到不在子路径中的点r,使它到子路径中的点的最大距离是所有不在子路径中的点到子路径中的点的最大距离中最大的。这个距离是连接这两类点的最大距离

Step 4:插入。在子路径中找到这样一条边(ij),使Distance[i][r]+Distance[r][j]-Distance[i][j]最小。将r插入到ij中间

Step 5:执行Step 3,直到所有点都加入到了路径中

4.5 凸包插入(connex hull insertion

时间复杂度:O(n^2*lg(n))

Step 1:构造凸包,并将它作为最初的子路径

Step 2对于所有不在子路径中的点r,找到其相应的在子路径中的点i,j,使得Distance[i][r]+Distance[r][j]-Distance[i][j]最小

Step 3:对于Step 2中找到的所有(ij0r),找到使(Distance[i][r]+Distance[r][j]/Distance[i][j]最小的那一组,将这组的r插入到ij中间

Step 4:回到Step 2,直到所有的点都加入到路径中

解释:    凸包(connex hull)是包含所有点的最小的多边形

如图:

5 双生成树算法(double spanning tree algorithm

时间复杂度:O(n^2)

       Step 1:由城市的点集构造最小生成树

       Step 2:重复所有的边,由此容易构造出Euler回路

       Step 3:去除Euler回路中的重复点,形成回路解

6 Christofides 算法

时间复杂度:O(n^3)

Christofides 算法由双生成树算法改进而来,使最坏解在最优解的1.5倍之内,而双生成树算法的最坏解在最优解的2倍之内

      

       Step 1:由城市的点集构造最小生成树MST

       Step 2:在最小生成树的奇顶点创建最小权匹配(minimum-weight matching ,MWM),将MWM添加到MST

       Step 3:从合并后的图中构造Euler回路

       Step 4:去除Euler回路中的重复点,形成回路解

这篇关于TSP 路径构造算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/999733

相关文章

随想录 Day 69 并查集 107. 寻找存在的路径

随想录 Day 69 并查集 107. 寻找存在的路径 理论基础 int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好vector<int> father = vector<int> (n, 0); // C++里的一种数组结构// 并查集初始化void init() {for (int i = 0; i < n; ++i) {father[i] = i;}

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

人工智能机器学习算法总结神经网络算法(前向及反向传播)

1.定义,意义和优缺点 定义: 神经网络算法是一种模仿人类大脑神经元之间连接方式的机器学习算法。通过多层神经元的组合和激活函数的非线性转换,神经网络能够学习数据的特征和模式,实现对复杂数据的建模和预测。(我们可以借助人类的神经元模型来更好的帮助我们理解该算法的本质,不过这里需要说明的是,虽然名字是神经网络,并且结构等等也是借鉴了神经网络,但其原型以及算法本质上还和生物层面的神经网络运行原理存在

SQL Server中,用Restore DataBase把数据库还原到指定的路径

restore database 数据库名 from disk='备份文件路径' with move '数据库文件名' to '数据库文件放置路径', move '日志文件名' to '日志文件存放置路径' Go 如: restore database EaseWe from disk='H:\EaseWe.bak' with move 'Ease

大林 PID 算法

Dahlin PID算法是一种用于控制和调节系统的比例积分延迟算法。以下是一个简单的C语言实现示例: #include <stdio.h>// DALIN PID 结构体定义typedef struct {float SetPoint; // 设定点float Proportion; // 比例float Integral; // 积分float Derivative; // 微分flo

将一维机械振动信号构造为训练集和测试集(Python)

从如下链接中下载轴承数据集。 https://www.sciencedirect.com/science/article/pii/S2352340918314124 import numpy as npimport scipy.io as sioimport matplotlib.pyplot as pltimport statistics as statsimport pandas

LeetCode 算法:二叉树的中序遍历 c++

原题链接🔗:二叉树的中序遍历 难度:简单⭐️ 题目 给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。 示例 1: 输入:root = [1,null,2,3] 输出:[1,3,2] 示例 2: 输入:root = [] 输出:[] 示例 3: 输入:root = [1] 输出:[1] 提示: 树中节点数目在范围 [0, 100] 内 -100 <= Node.

【Java算法】滑动窗口 下

​ ​    🔥个人主页: 中草药 🔥专栏:【算法工作坊】算法实战揭秘 🦌一.水果成篮 题目链接:904.水果成篮 ​ 算法原理 算法原理是使用“滑动窗口”(Sliding Window)策略,结合哈希表(Map)来高效地统计窗口内不同水果的种类数量。以下是详细分析: 初始化:创建一个空的哈希表 map 用来存储每种水果的数量,初始化左右指针 left

ROS2从入门到精通4-4:局部控制插件开发案例(以PID算法为例)

目录 0 专栏介绍1 控制插件编写模板1.1 构造控制插件类1.2 注册并导出插件1.3 编译与使用插件 2 基于PID的路径跟踪原理3 控制插件开发案例(PID算法)常见问题 0 专栏介绍 本专栏旨在通过对ROS2的系统学习,掌握ROS2底层基本分布式原理,并具有机器人建模和应用ROS2进行实际项目的开发和调试的工程能力。 🚀详情:《ROS2从入门到精通》 1 控制插