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

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

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

这一步骤使用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. bzoj1603 / P2913 [USACO08OCT]车轮旋转Wheel Rotation

    P2913 [USACO08OCT]车轮旋转Wheel Rotation 稳妥起见(防止数据出锅),用了bfs 每次的转移可以直接用异或和解决. #include<iostream> #i ...

  2. 在apache中使用.htaccess文件的注意事项

    在apache的配置文件中: <VirtualHost *:80> ServerName tp5.com DocumentRoot d:/wamp/www/tp5.com/public & ...

  3. Windows10系统远程桌面连接出现卡顿如何解决

    最新的windows10系统下,用户只要开启远程桌面连接,就能够轻松地操控其他电脑.但是,最近部分用户在win10中启用远程连接时,发现电脑窗口变得非常缓慢卡顿,这是怎么回事呢?其实,该问题与系统的设 ...

  4. 解决方案--duilib中edit获得鼠标焦点后右边框被覆盖

    用duilib做了一个登录框,用户名的输入使用的是edit控件. XML中是这样写的: <Edit name="subdomain_edit" tipvalue=" ...

  5. 使用 v-cloak 防止页面加载时出现 vue.js 的变量名

    知识点:使用 v-cloak 防止页面加载时出现 vue.js 的变量名 场景:在使用vue语法,实现下拉框功能时,展示数据列表之前,出现对应的 vuejs 变量名 代码: var vm = new ...

  6. centos 升级python26到python27

    由于开发库依赖于python27,而自己安装的centos6.8自带的python是2.6.6,因此打算简单的做一下升级. 因为centos的yum依赖于python26因此不打算覆盖26.步骤如下: ...

  7. linux文件锁的应用,POSIX,unix标准,linux标准

    1. perl,flock加锁.java也能加锁. 2. 先创建文件并打开,才能加锁(写打开?). 3. 可以用于判断进程是否一直在运行(用另一进程判断),如果锁一直在,则进程在:锁不在,则原进程或意 ...

  8. XML_CPP_资料_libXml2_01

    ZC: 看了一些 C/C++的XML文章,也看了一些 Qt的 QXmlQuery/QXmlSimpleReader/QXmlStreamReader/QXmlStreamWriter 的文章.总体感觉 ...

  9. QWebEngineView_简单例子_01

    工程名 : WebEngine01 1.WebEngine01.pro #------------------------------------------------- # # Project c ...

  10. linux 查看日志最后几行

    tail -n 50 wx.log 示例:查看/var/log/boot.log,只显示最后一行.则执行 tail -n 1  /var/log/boot.log tail -n 1000:显示最后1 ...