news 2026/6/15 15:30:39

旅行商问题(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/6/16 5:25:53

Magpie-LuckyDraw:多平台3D抽奖系统的技术架构深度解析

Magpie-LuckyDraw&#xff1a;多平台3D抽奖系统的技术架构深度解析 【免费下载链接】Magpie-LuckyDraw &#x1f3c5;A fancy lucky-draw tool supporting multiple platforms&#x1f4bb;(Mac/Linux/Windows/Web/Docker) 项目地址: https://gitcode.com/gh_mirrors/ma/Magp…

作者头像 李华
网站建设 2026/6/15 4:31:07

数据治理如何真正落地?这8大案例的破局之战,就是你的避坑指南

我们都知道数据治理很重要&#xff0c;但一提到如何落地&#xff0c;很多人都会陷入“道理都懂&#xff0c;却依然做不好”的困境。真正的难点在于&#xff0c;不同行业、不同规模、不同痛点的企业&#xff0c;该如何找到那条专属的实施路径&#xff1f;亿信华辰新书《数据治理…

作者头像 李华
网站建设 2026/6/15 21:31:20

EmotiVoice在语音广告制作中的高效应用案例

EmotiVoice在语音广告制作中的高效应用 在数字营销的浪潮中&#xff0c;品牌与用户之间的每一次触达都变得愈发珍贵。尤其是在短视频、社交媒体和电商平台主导流量入口的今天&#xff0c;一条30秒的语音广告可能决定一次冲动消费是否发生。然而&#xff0c;传统语音广告制作却仍…

作者头像 李华
网站建设 2026/6/13 19:51:10

科技不应逾越人性底线:我们的立场声明

科技不应逾越人性底线&#xff1a;我们的立场声明 在某次深夜调试语音助手时&#xff0c;我听到一段由AI生成的“愤怒”语音——语速急促、音调尖锐&#xff0c;几乎与真人无异。那一刻&#xff0c;我没有感到技术突破的欣喜&#xff0c;反而心头一紧&#xff1a;如果这声音被用…

作者头像 李华
网站建设 2026/6/15 2:17:42

Lime开源编辑器深度体验:从Sublime Text用户到贡献者的完整解析

Lime开源编辑器深度体验&#xff1a;从Sublime Text用户到贡献者的完整解析 【免费下载链接】lime Open source API-compatible alternative to the text editor Sublime Text 项目地址: https://gitcode.com/gh_mirrors/li/lime 作为一名长期使用Sublime Text的开发者&…

作者头像 李华
网站建设 2026/6/15 13:41:02

ThingsBoard物联网平台Vue3前端开发实战指南

ThingsBoard物联网平台Vue3前端开发实战指南 【免费下载链接】thingsboard-ui-vue3 本项目为基于Vue3开发的 ThingsBoard 前台 ,AntDesginVue、VbenVueAdmin、AntV X6、规则链代码已全部开放、ThingsBoard3.x持续更新中 项目地址: https://gitcode.com/oliver225/thingsboard…

作者头像 李华