news 2026/5/1 9:41:31

旅行商问题(TSP) C++ 解法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
旅行商问题(TSP) C++ 解法

旅行商问题(TSP)是经典的组合优化问题,核心是给定 n 个城市及城市间的距离,求访问所有城市一次且仅一次后回到起点的最短路径。这里介绍两种易理解的解法:暴力枚举法(适合小规模数据)和动态规划法(适合中等规模数据)。

一、 暴力枚举法

原理

枚举所有城市的排列组合,计算每种排列的路径长度,取最小值。n 个城市的排列数为 (n-1)! (起点固定,减少重复计算)。

C++ 代码

#include <iostream>

#include <vector>

#include <algorithm>

#include <climits>

using namespace std;

// 计算路径总长度

int calculatePath(const vector<vector<int>>& dist, const vector<int>& path) {

int total = 0;

int n = path.size();

for (int i = 0; i < n - 1; i++) {

total += dist[path[i]][path[i + 1]];

}

total += dist[path[n - 1]][path[0]]; // 回到起点

return total;

}

int TSP_brute(const vector<vector<int>>& dist) {

int n = dist.size();

vector<int> path(n);

for (int i = 0; i < n; i++) path[i] = i; // 初始化路径 0->1->2->...->n-1

int min_dist = INT_MAX;

// 固定起点为0,枚举其余城市的排列

do {

if (path[0] != 0) break;

int current = calculatePath(dist, path);

if (current < min_dist) {

min_dist = current;

}

} while (next_permutation(path.begin(), path.end()));

return min_dist;

}

int main() {

// 邻接矩阵表示城市距离,dist[i][j]是城市i到j的距离

vector<vector<int>> dist = {

{0, 10, 15, 20},

{10, 0, 35, 25},

{15, 35, 0, 30},

{20, 25, 30, 0}

};

int min_path = TSP_brute(dist);

cout << "最短路径长度:" << min_path << endl;

return 0;

}

优缺点

优点:逻辑简单,容易理解和实现。

缺点:时间复杂度极高 O((n-1)!) ,n>10 时运行效率极低。

二、 动态规划法

原理

基于状态压缩 DP,用 dp[mask][u] 表示访问过的城市集合为 mask,当前位于城市 u 时的最短路径长度。

mask 是二进制数,第 i 位为 1 表示城市 i 已访问。

初始状态: dp[1<<0][0] = 0 (只访问起点 0,路径长度为 0)。

状态转移: dp[mask][u] = min(dp[mask ^ (1<<u)][v] + dist[v][u]) (v 是 mask 中已访问的城市)。

最终答案: min(dp[(1<<n)-1][u] + dist[u][0]) (访问所有城市后回到起点)。

C++ 代码

#include <iostream>

#include <vector>

#include <climits>

using namespace std;

const int INF = INT_MAX / 2; // 防止加法溢出

int TSP_dp(const vector<vector<int>>& dist) {

int n = dist.size();

int max_mask = 1 << n;

// dp[mask][u]: 状态mask,当前在u的最短路径

vector<vector<int>> dp(max_mask, vector<int>(n, INF));

dp[1 << 0][0] = 0; // 初始状态

for (int mask = 1; mask < max_mask; mask++) {

for (int u = 0; u < n; u++) {

if (!(mask & (1 << u))) continue; // u不在mask中,跳过

// 枚举上一个访问的城市v

for (int v = 0; v < n; v++) {

if (u == v || !(mask & (1 << v))) continue;

dp[mask][u] = min(dp[mask][u], dp[mask ^ (1 << u)][v] + dist[v][u]);

}

}

}

// 访问所有城市后回到起点0

int min_dist = INF;

int full_mask = (1 << n) - 1;

for (int u = 1; u < n; u++) {

min_dist = min(min_dist, dp[full_mask][u] + dist[u][0]);

}

return min_dist;

}

int main() {

vector<vector<int>> dist = {

{0, 10, 15, 20},

{10, 0, 35, 25},

{15, 35, 0, 30},

{20, 25, 30, 0}

};

int min_path = TSP_dp(dist);

cout << "最短路径长度:" << min_path << endl;

return 0;

}

优缺点

优点:时间复杂度 O(n²×2ⁿ) ,比暴力法高效,能处理 n≤20 的数据。

缺点:空间复杂度 O(n×2ⁿ) ,n 过大时内存消耗大。

三、 测试与总结

1. 上述代码的测试用例是 4 个城市,最短路径长度为 80。

2. 暴力法适合理解 TSP 问题本质,动态规划法是入门级高效解法。

3. 对于更大规模数据,可学习贪心算法或遗传算法等近似解法。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/24 5:09:07

Phi-3-Mini-4K-Instruct:3步快速上手的轻量级AI模型安装指南

Phi-3-Mini-4K-Instruct&#xff1a;3步快速上手的轻量级AI模型安装指南 【免费下载链接】Phi-3-mini-4k-instruct-gguf 项目地址: https://ai.gitcode.com/hf_mirrors/ai-gitcode/Phi-3-mini-4k-instruct-gguf 还在为复杂的AI模型安装而头疼吗&#xff1f;&#x1f91…

作者头像 李华
网站建设 2026/4/30 16:29:23

明星声音被模仿?EmotiVoice防滥用机制说明

明星声音被模仿&#xff1f;EmotiVoice防滥用机制说明 在AI语音技术飞速发展的今天&#xff0c;一段几秒钟的音频就能“复制”出某位明星的声音——这不再是科幻电影的情节。从虚拟偶像直播带货&#xff0c;到智能助手模仿亲人语调安慰用户&#xff0c;语音合成正变得越来越真实…

作者头像 李华
网站建设 2026/4/20 23:11:27

准确--CentOS 7 配置用户资源限制(nofile / nproc)

CentOS 7 配置用户资源限制&#xff08;nofile / nproc&#xff09;目标&#xff1a; 提升文件句柄数&#xff08;nofile&#xff09;提升进程数&#xff08;nproc&#xff09;避免启用 UsePAM 后“密码正确却登录失败”的坑一、临时生效&#xff08;立即生效&#xff0c;无需重…

作者头像 李华
网站建设 2026/4/17 15:53:30

WarcraftHelper:魔兽争霸III终极兼容性修复工具完整指南

WarcraftHelper&#xff1a;魔兽争霸III终极兼容性修复工具完整指南 【免费下载链接】WarcraftHelper Warcraft III Helper , support 1.20e, 1.24e, 1.26a, 1.27a, 1.27b 项目地址: https://gitcode.com/gh_mirrors/wa/WarcraftHelper WarcraftHelper是一款专为解决魔兽…

作者头像 李华
网站建设 2026/4/18 8:17:08

Labelme终极指南:从零开始掌握图像标注全流程

Labelme终极指南&#xff1a;从零开始掌握图像标注全流程 【免费下载链接】labelme Image Polygonal Annotation with Python (polygon, rectangle, circle, line, point and image-level flag annotation). 项目地址: https://gitcode.com/gh_mirrors/la/labelme 还在为…

作者头像 李华
网站建设 2026/4/26 5:58:03

Wan2GP视频生成教程:从零开始掌握AI视频创作

Wan2GP视频生成教程&#xff1a;从零开始掌握AI视频创作 【免费下载链接】Wan2GP Wan 2.1 for the GPU Poor 项目地址: https://gitcode.com/gh_mirrors/wa/Wan2GP Wan2GP是一款强大的开源视频生成工具&#xff0c;基于Wan2.1模型构建&#xff0c;专为GPU资源有限的用户…

作者头像 李华