状态压缩 dp: 最短Hamilton路径
c++
最短Hamilton路径
/* 题目描述: 背景介绍: 给定一张 n 个点的带权无向图,点从 0∼n−1 标号,求起点 0 到终点 n−1 的最短 Hamilton 路径。 Hamilton 路径的定义是从 0 到 n−1 不重不漏地经过每个点恰好一次。 输入格式: 第一行输入整数 n。 接下来 n 行每行 n 个整数,其中第 i 行第 j 个整数表示点 i 到 j 的距离(记为 a[i,j])。 对于任意的 x,y,z,数据保证 a[x,x]=0,a[x,y]=a[y,x] 并且 a[x,y]+a[y,z]≥a[x,z]。 数据范围: 1 ≤ n ≤ 20 0 ≤ a[i,j] ≤ 10^7 求解思路: 状态定义: f[i][state] 表示当前经过路径为 state, state 二进制形式表示了当前路径所经过的点,其中 i 点,是当前所在点。 在 state 的路径的基础上,可以继续扩充,扩展到其他点 j 上,同时 state 更新 -> state2 复杂度: O(N 2 ^ N) */ #include <iostream> #include <cstring> #include <algorithm> #include <cstdio> #include <queue> using namespace std; const int N = 20, INF = 0x3f3f3f3f; int f[1 << N][N]; int n; int w[N][N]; int main() { scanf("%d", &n); for (int i = 0; i < n; i ++ ) { for (int j = 0; j < n; j ++ ) { scanf("%d", &w[i][j]); } } memset(f, 0x3f, sizeof f); // 这个是表明从 0 点开始 f[1][0] = 0; for (int state = 0; state < (1 << n); state ++ ) { for (int u = 0; u < n; u ++ ) { if (f[state][u] == INF) { continue; } if ((state >> u) & 1) { for (int v = 0; v < n; v ++ ) { if (((state >> v) & 1) == 0) { f[state | (1 << v)][v] = min(f[state | (1 << v)][v], f[state][u] + w[u][v]); } } } } } printf("%d\n", f[(1 << n) - 1][n - 1]); return 0; }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)