news 2026/5/16 23:05:46

旅行商问题(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/5/12 10:06:53

我做了一个「人生 K 线」工具:不是预测,而是阶段理解

最近完成了一个个人项目&#xff0c;想在 CSDN 记录一下整体设计思路。 PredictorsGPT.comhttps://www.predictorsgpt.com/ 这个项目可以简单理解为一个英文版的「人生 K 线」工具&#xff0c;核心目的不是预测未来&#xff0c;而是帮助用户理解自己所处的人生阶段和节奏。 一…

作者头像 李华
网站建设 2026/5/15 12:28:41

Browserpass浏览器扩展完整使用指南:安全密码管理三步走

Browserpass浏览器扩展完整使用指南&#xff1a;安全密码管理三步走 【免费下载链接】browserpass-extension Browserpass web extension 项目地址: https://gitcode.com/gh_mirrors/br/browserpass-extension Browserpass是一款专为pass密码管理器设计的浏览器扩展工具…

作者头像 李华
网站建设 2026/5/16 14:24:35

3倍提速!MiniGPT-4批量推理优化实战指南

3倍提速&#xff01;MiniGPT-4批量推理优化实战指南 【免费下载链接】MiniGPT-4 Open-sourced codes for MiniGPT-4 and MiniGPT-v2 (https://minigpt-4.github.io, https://minigpt-v2.github.io/) 项目地址: https://gitcode.com/gh_mirrors/mi/MiniGPT-4 MiniGPT-4作…

作者头像 李华
网站建设 2026/5/13 8:48:40

EmotiVoice赋能有声书制作:自动生成带情节情绪的朗读

EmotiVoice赋能有声书制作&#xff1a;自动生成带情节情绪的朗读 在有声内容爆发式增长的今天&#xff0c;听众早已不满足于“能听就行”的机械朗读。一部真正打动人心的有声书&#xff0c;需要语气的起伏、情感的流动&#xff0c;甚至角色性格的细微差别——这些原本只能由专业…

作者头像 李华
网站建设 2026/5/16 14:39:16

情人节专属:用爱人声音生成甜蜜告白

情人节专属&#xff1a;用爱人声音生成甜蜜告白 在智能语音助手每天机械地报天气、设闹钟的今天&#xff0c;你有没有想过——它也能轻声说一句&#xff1a;“亲爱的&#xff0c;我想你了”&#xff1f;而且&#xff0c;是用你爱人的声音。 这听起来像科幻电影的情节&#xff0…

作者头像 李华
网站建设 2026/5/11 9:27:12

让AI语音成为桥梁,而不是替代

让AI语音成为桥梁&#xff0c;而不是替代 在虚拟助手机械地念出天气预报、有声书用千篇一律的语调讲述跌宕起伏的故事时&#xff0c;我们不禁要问&#xff1a;为什么技术越先进&#xff0c;声音反而越“不像人”&#xff1f; 语音的本质是情感的载体。人类交流中超过70%的信息…

作者头像 李华