题目的大概意思就是一个人到一些城市送披萨,要求找到一条路径可以遍历每个城市后返回出发点,而且路径距离最短。最后输出最短距离就可以。

注意:每个城市可反复訪问多次。

因为题中明白说了两个城市间的直接可达路径(即不经过其他城市结点)不一定是最短路径。所以须要借助邻接矩阵首先求出随意两个城市间的最短距离。

这一步骤使用Floyd最短路径算法就可以。

然后,在此基础上来求出遍历各个城市后回到出发点的最短路径的距离,即求解TSP问题。

TSP问题眼下有多种解法:搜索解法,动归解法。启示式解法。这里就针对poj 3311问题给出了前两种解法。

搜索解法:这样的解法事实上就是计算排列子集树的过程。从0点出发。要求遍历1。2,3点后回到0点。

以不同的顺序来依次遍历1,2,3点就会导出不同的路径(0->1->2->3->0;0->1->3->2->0等等),总共同拥有3!=6条路径须要考虑,从中选出最短的那条就是所求。搜索解法的时间复杂度为O(n!)

附上搜索代码:

#include <iostream>
#include <vector>
#include <algorithm>
#include <utility>
#include <cstdio>
#include <cstdlib> using namespace std; int n;
vector<vector<int> > links;
vector<vector<int> > sp;
vector<bool> used;
long long ans; void Floyed()
{
sp = links;
for(int k = 0; k < n; ++k)
{
for(int i = 0; i < n; ++i)
{
for(int j = 0; j < n; ++j)
sp[i][j] = min(sp[i][j], sp[i][k] + sp[k][j]);
}
}
//for(int i = 0; i < n; ++i)
//{
//for(int j = 0; j < n; ++j)
//cout << sp[i][j] << " ";
//cout << endl;
//}
} void Backtrack(int level, int v, long long cost)
{
if( level == n - 1 )
{
ans = min(cost + sp[v][0], ans);
return;
} for(int i = 0; i < n; ++i)
{
if( !used[i] )
{
used[i] = true;
Backtrack(level + 1, i, cost + sp[v][i]);
used[i] = false;
}
}
} void Work()
{
Floyed();
ans = 1e8;
used.assign(n, false);
used[0] = true;
Backtrack(0, 0, 0);
//cout << "ans = ";
cout << ans << endl;
} int main()
{
//freopen("3311.tst", "r", stdin);
while( cin >> n && n )
{
++n;
//links.resize(n, vector<int>(n)); 将这一句替换为以下这一句。就会WA,还请高手可以不吝赐教!
links.assign(n, vector<int>(n, 0));
for(int i = 0; i < n; ++i)
{
for(int j = 0; j < n; ++j)
cin >> links[i][j];
}
Work();
}
return 0;
}

动归解法:细致观察搜索解法的过程,事实上是有非常多反复计算的。比方从0点出发。经过1,2。3,4,5点后回到0点。那么0->1->2->(3。4。5三个点的排列)->0与0->2->1->(3,4,5三个点的排列)->0就存在反复计算(3。4,5三点的排列)->0路径集上的最短路径。仅仅要我们可以将这些状态保存下来就行减少一部分复杂度。以下就让我们用动归来求解这一问题。记dp(v, S)为从v点出发,遍历S集合中的每个点后,回到出发点(0点)的最短距离。递推表达式的推导例如以下:

假设S为空集,即没有须要遍历的结点了。此时可以直接从v点回到0点。则dp(v,S)=sp[v][0] //sp[v][0]是v点到0点的最短路径距离

假设S不为空集。则dp(v,S)=min{sp[v][u] + dp(v,S-{u})}//sp[v][u]是v点到u点的最短路径距离

上述过程怎样用编码实现呢,主要难点就在于集合S的表示。我们可以用位比特来表示一个集合。如集合{1,2,3}。{1,2}分别可以用7(111),3(011)来表示。

所以动归整个状态二维表的大小为n*2^n。而表中的每个元素的计算须要O(n)的复杂度,所以动态规划的时间复杂度为O(n^2*2^n)

附上动态规划代码:

#include <iostream>
#include <vector>
#include <algorithm>
#include <utility>
#include <cstdio>
#include <cstdlib> using namespace std; int n;
vector<vector<int> > links;
vector<vector<int> > sp;
vector<bool> used;
vector<vector<long long> > dp;
long long ans; void Floyed()
{
sp = links;
for(int k = 0; k < n; ++k)
{
for(int i = 0; i < n; ++i)
{
for(int j = 0; j < n; ++j)
sp[i][j] = min(sp[i][j], sp[i][k] + sp[k][j]);
}
}
//for(int i = 0; i < n; ++i)
//{
//for(int j = 0; j < n; ++j)
//cout << sp[i][j] << " ";
//cout << endl;
//}
} long long CalcVal(int v, long long bit)
{
if( dp[v][bit] != -1 )
{
return dp[v][bit];
}
if( !bit )
{
dp[v][bit] = sp[v][0];
}
else
{
long long ret = 1e8;
for(int i = 1; i < n; ++i)
{
int b = 1 << i - 1;
if( b&bit )
{
ret = min(ret, sp[v][i] + CalcVal(i, b-bit));
}
}
dp[v][bit] = ret;
}
return dp[v][bit];
} void Work()
{
Floyed();
long long m = (1 << n - 1) - 1;
dp.assign(n, vector<long long>(m, -1));
ans = 1e8;
for(int i = 1; i < n; ++i)
{
long long b = 1 << i - 1;
ans = min(ans, sp[0][i] + CalcVal(i, b-m));
}
//cout << "ans = ";
cout << ans << endl;
} int main()
{
//freopen("3311.tst", "r", stdin);
while( cin >> n && n )
{
++n;
//links.resize(n, vector<int>(n));
links.assign(n, vector<int>(n, 0));
for(int i = 0; i < n; ++i)
{
for(int j = 0; j < n; ++j)
cin >> links[i][j];
}
Work();
}
return 0;
}

poj3311 经典tsp问题的更多相关文章

  1. [状压dp]经典TSP

    0出发 每个顶点经过一次 回到0 最小花费. O($n^2 \times 2^n$) 记忆化搜索: // s: 已经访问过的节点状态 v: 出发位置 int dfs(int s, int v) { ) ...

  2. POJ3311 Hie with the Pie(状压DP,Tsp)

    本题是经典的Tsp问题的变形,Tsp问题就是要求从起点出发经过每个节点一次再回到起点的距离最小值,本题的区别就是可以经过一个节点不止一次,那么先预处理出任意两点之间的最短距离就行了,因为再多走只会浪费 ...

  3. HDU - 5067 / HDU - 5418 TSP

    集合表示多用[0,n)表示方法 HDU - 5067 经典TSP,每个顶点恰经过一次最优 #include<bits/stdc++.h> #define rep(i,j,k) for(in ...

  4. Hie with the Pie(poj3311)

    题目链接:http://poj.org/problem?id=3311 学习博客:https://blog.csdn.net/u013480600/article/details/19692985 H ...

  5. poj 3311 状压DP

    经典TSP变形 学到:1.floyd  O(n^3)处理随意两点的最短路 2.集合的位表示,我会在最后的总结出写出.注意写代码之前一定设计好位的状态.本题中,第0位到第n位分别代表第i个城市,1是已经 ...

  6. 状态压缩DP(大佬写的很好,转来看)

    奉上大佬博客 https://blog.csdn.net/accry/article/details/6607703 动态规划本来就很抽象,状态的设定和状态的转移都不好把握,而状态压缩的动态规划解决的 ...

  7. (转)POJ题目分类

    初期:一.基本算法:     (1)枚举. (poj1753,poj2965)     (2)贪心(poj1328,poj2109,poj2586)     (3)递归和分治法.     (4)递推. ...

  8. poj分类

    初期: 一.基本算法:      (1)枚举. (poj1753,poj2965)      (2)贪心(poj1328,poj2109,poj2586)      (3)递归和分治法.      ( ...

  9. 转载 ACM训练计划

    leetcode代码 利用堆栈:http://oj.leetcode.com/problems/evaluate-reverse-polish-notation/http://oj.leetcode. ...

随机推荐

  1. Common Knowledge

    2014-08-23 11:01:11 -6:四面体 (1)内切球半径:r = 3V / (S1+S2+S3+S4) (2)体积:将四点组成三个向量AB,AC,AD,向量的混合积就是它们组成的平行六面 ...

  2. eclipse中gradle插件安装

    help===>install software===>http://download.eclipse.org/buildship/updates/e46/releases/2.x/

  3. SQL 中 not in 查询不到数据问题

    在开发的过程中,遇到过not in 始终查询不到数据问题 select * from T_CustomerInfo where CustomerID not in (select CustomerID ...

  4. HDU 6425 Rikka with Badminton(组合问题签到)题解

    题意:问你有多少种选法使得不能满足大于等于2个拍子且大于等于1个球. 思路:数学组合问题,分类讨论一下,刚开始的时候分的很乱,写的乱七八糟的...还有注意MOD,基本上有大数相乘的地方都要先MOD一下 ...

  5. BZOJ 2226 【SPOJ 5971】 LCMSum

    题目链接:LCMSum 这个题显然就是要我们推式子了……那么就来推一波: \begin{aligned}&\sum_{i=1}^n lcm(i,n) \\=&\sum_{i=1}^n\ ...

  6. a标签 在新页面打开

    <a href="https://www.baidu.com/" target="_blank">下载</a>

  7. Vue.js的类Class 与属性 Style如何绑定

    Vue.js的类Class 与属性 Style如何绑定 一.总结 一句话总结:数据绑定一个常见需求是操作元素的 class 列表和它的内联样式.因为它们都是属性,我们可以用 v-bind 处理它们:我 ...

  8. Python 中的那些坑总结——持续更新

    1.三元表达式之坑 很显然,Python把第一行的(10 + 4)看成了三元表达式的前部分,这个坑是看了<Python cookbook>(P5)中学到的,书中的代码: 2.Python生 ...

  9. css3伪放大镜(图片放大动画)效果(鼠标移入圆形区域放大图片)

    源码: <!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8&q ...

  10. [转载]SQL语句练习

    .查询“生物”课程比“物理”课程成绩高的所有学生的学号: 思路: 获取所有有生物课程的人(学号,成绩) - 临时表 获取所有有物理课程的人(学号,成绩) - 临时表 根据[学号]连接两个临时表: 学号 ...