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

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

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

这一步骤使用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. Python基础笔记之同时装了Python3和Python2,怎么在命令行使用pip

    我们在安装Python3(>=3.3)时,Python的安装包实际上在系统中安装了一个启动器py.exe,默认放置在文件夹C:\Windows\下面.这个启动器允许我们指定使用Python2还是 ...

  2. 使用MongoVUE无法添加Collection

    说明: 问题MongoDB版本为3.2,MongoVUE 1.6.9 问题: 在数据库中添加集合不可用. 解决方法: MongoDB版本换成2 或者 切换存储引擎: 从MongoDB 3.2 版本开始 ...

  3. 辅助模块应用(auxiliary/scanner/portscan/tcp)

    实验步骤 创建msf所需的数据库 之前我们开启msf时下面总会出现一个红色的小减号,原来是因为没有和数据库键连接,于是首先我们要手动建立一个数据库... 使用命令来实现: service postgr ...

  4. 20145219《网络对抗》Web基础

    20145219<网络对抗>Web基础 基础问题回答 什么是表单? HTML表单用于收集用户输入,用<form>元素定义,包含不同类型的 input元素.复选框.单选按钮.提交 ...

  5. 20145219《网络对抗技术》PC平台逆向破解之逆向与Bof基础

    20145219<网络对抗技术>PC平台逆向破解之逆向与Bof基础 实践目标 实践对象:一个名为pwn1的linux可执行文件. pwn1正常执行流程:main调用foo函数,foo函数会 ...

  6. Android Studio安装与使用

    2013年谷歌推出android studio后,单独支持android开发,这是基于Java语言集成开发环境IntelliJ搭建的IDE.特别在android studio1.0稳定版出来后,谷歌将 ...

  7. ubuntu16.04下安装TensorFlow(GPU加速)----详细图文教程【转】

    本文转载自:https://blog.csdn.net/zhaoyu106/article/details/52793183 le/details/52793183 写在前面 一些废话 接触深度学习已 ...

  8. vim 录制宏,自动循环执行组合操作

    前言 在 vim 中录制宏的意思就是把一段操作录制下来,然后可以重复执行.打个比方,我有一个文本,一共 50000 行,我要在每一行后面加一个英文句号.这个操作如果手动做的话效率非常低.这时候我可以这 ...

  9. P4 PI库安装

    repo:Github PI P4Runtime - a control plane framework and tools for the P4 programming language 这个是P4 ...

  10. Linux 用户和组的 添加/删除

    1.建用户:adduser phpq //新建phpq用户passwd phpq //给phpq用户设置密码 2.建工作组groupadd test //新建test工作组 3.新建用户同时增加工作组 ...