code2800 送外卖
首先,对图进行一次Floyd(g[][]是图)
1.dfs:(u是当前在的节点,d是已经走的路程)
void dfs(int u,int d){
if(d>=ans)return;
bool flag=false;
for(int i=;i<=n;i++){
if(vis[i]==false&&i!=u){
flag=true;
vis[i]=true;
dfs(i,d+g[u][i]);
vis[i]=false;
}
}
if(flag==false){
d+=g[u][];
if(ans>d)ans=d;
}
}
起初我认为n<=15是dfs完全可以解决的,但是忽略了一个问题:每对城市之间都可以互相到达,边特别多,结果TLE ^_^;
正解是压缩状态的dp:
dp[u][s]表示状态:走到u这个点并且前面已经走完了集合s中的点 的最短距离
s是二进制压缩状态 1到过 0没有
ps:u这个点本身也包括在集合s中
s0是把s中u去掉后的集合 s0 = s & ((1<<n)-1 - (1<<(u-1)) ) //@_@a...
dp[u][s]=min{ dp[i][s0]+g[i][u] }
i是集合s0中包含的点,注意是 +g[i][u],因为是先走到i,再从i到u(题目中说城市a到城市b的时间和城市b到城市a的时间不一定相同)
代码如下:
#include<iostream>
#include <bitset>
#include<cstring>
using namespace std; int n;
int g[][];
int dp[][<<];
bool vis[][<<]; void Floyd(){
for(int k=;k<=n;k++){
for(int i=;i<=n;i++){
for(int j=;j<=n;j++){
if(i==k||j==k||i==j)continue;
g[i][j]=min(g[i][j], g[i][k]+g[k][j]);
}
}
}
} /*void dfs(int u,int d){
if(d>=ans)return; bool flag=false;
for(int i=1;i<=n;i++){
if(vis[i]==false&&i!=u){
flag=true;
vis[i]=true;
dfs(i,d+g[u][i]);
vis[i]=false;
}
}
if(flag==false){
d+=g[u][0];
if(ans>d)ans=d; }
}*/ int dfs(int u,int s){
if(vis[u][s])return dp[u][s];
vis[u][s]=true; int s0= s & ((<<n)- - (<<(u-)) );
//cout<<u<<" "<<bitset<sizeof(int)*8>(s)<<" "<<bitset<sizeof(int)*8>(s0)<<endl;
int cnt=;
int& ans=dp[u][s];
for(int i=;i<=;i++){
if(s0&(<<(i-))){
cnt++;
int x=dfs(i, s0)+g[i][u];
ans=min(ans,x);
}
}
if(cnt==){
ans=g[][u];
}
return ans;
} int main(){
//freopen("2800.txt","r",stdin);
memset(dp,0x3f,sizeof(dp));
memset(vis,false,sizeof(vis)); cin>>n;
for(int i=;i<=n;i++){
for(int j=;j<=n;j++){
cin>>g[i][j];
}
} Floyd();
/*for(int i=0;i<=n;i++){
for(int j=0;j<=n;j++){
cout<<g[i][j]<<' ';
}
cout<<endl;
}*/ cout<<dfs(,(<<n)-)<<endl; /*int a,b;
while(cin>>a>>b){
cout<<dp[a][b]<<endl;
}*/
//fclose(stdin);
return ;
}
ps:你看到了大量的注释,它们包括深搜算法和Debug(我懒得删去了^_^!)
ps:文字表情太帅了! 来一个最经典的 :)
code2800 送外卖的更多相关文章
- [codevs 2800]送外卖
题目描述 Description 有一个送外卖的,他手上有n份订单,他要把n份东西,分别送达n个不同的客户的手上.n个不同的客户分别在1~n个编号的城市中.送外卖的从0号城市出发,然后n个城市都要走一 ...
- codevs2800 送外卖
题目描述 Description 有一个送外卖的,他手上有n份订单,他要把n份东西,分别送达n个不同的客户的手上.n个不同的客户分别在1~n个编号的城市中.送外卖的从0号城市出发,然后n个城市都要走一 ...
- 【CodeVS2800】 送外卖 最短路+状压DP
首先求出各点之间的最短路,floyed即可,注意是0-n. 然后考虑状压,f[i][j]表示状态为i时访问j点时的最短路和,1表示访问,0表示未访问,然后第j个点所在的位置就是(1<<j) ...
- 【wikioi】2800 送外卖(状压dp+floyd)
http://www.wikioi.com/problem/2800/ 本题状压莫名其妙的tle了,(按照hzwer大神打的喂,他1000多ms,我就2000ms了?) (14.8.7更,将getnu ...
- 从送外卖到建站售主机还有共享自行车说起-2017年8月江西IDC排行榜与发展报告
曾几何时,送外卖,这样的"低技术含量"工作,很难被互联网公司看上,直到百度将其当作连接终端用户与大数据的管道. 同样,销售主机域名和建站业务,本也是"微小体量" ...
- 2800 送外卖[状态压缩dp]
2800 送外卖 时间限制: 2 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 题解 题目描述 Description 有一个送外卖的,他手上有n份订单,他 ...
- codevs 2800 送外卖 TSP问题
2800 送外卖 时间限制: 2 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 题目描述 Description 有一个送外卖的,他手上有n份订单,他要把n份 ...
- 送外卖(codevs 2800)
题目描述 Description 有一个送外卖的,他手上有n份订单,他要把n份东西,分别送达n个不同的客户的手上.n个不同的客户分别在1~n个编号的城市中.送外卖的从0号城市出发,然后n个城市都要走一 ...
- CODEVS_2800 送外卖 状态压缩+动态规划
原题链接:http://codevs.cn/problem/2800/ 题目描述 Description 有一个送外卖的,他手上有n份订单,他要把n份东西,分别送达n个不同的客户的手上.n个不同的客户 ...
随机推荐
- Vue.js实现数据的双向数据流
众所周知,Vue.js一直使用的是单向数据流的,和angularJs的双向数据流相比,单向数据流更加容易控制.Vue.js允许父组件通过props属性传递数据到子组件.但是有些情况下我们需要在子组件里 ...
- 20181105_线程之Task
Task是基于.net Framework3.0框架, Task使用的线程也是来自于ThreadPool 多线程的两个意义: 优化体验(常见于不卡界面), 提升运行速度(不同线程可以分担运算任务) 总 ...
- 启动 Eclipse 报错 “An internal error occurred during: "Initializing Java Tooling". java.lang.NullPointerException”
之前在线升级了Eclipse,由于网络/或者是设置问题,在升级完成后启动Eclipse出线上述错误... 解决方法 1. 删除目录工作目录下面的.project文件夹: 如下图: 2. 关闭Eclip ...
- 非对称加密与OpenSSL
随着个人隐私越来越受重视, HTTPS也渐渐的流行起来, 甚至有许多网站都做到了全站HTTPS, 然而这种加密和信任机制也不断遭遇挑战,比如戴尔根证书携带私钥,Xboxlive证书私钥泻露, 还有前一 ...
- SVN的学习
SVN是Subversion的简称,是一个开放源代码的版本控制系统,相较于CVS,它采用了分支管理系统,它的设计目标就是取代CVS 为什么需要SVN 通常软件开发由多人协作开发,如果对代码文件.配置文 ...
- 【BZOJ】1926: [Sdoi2010]粟粟的书架(暴力+主席树)
题目 传送门:QWQ 分析 两道题目 第一问暴力预处理 用$ a[i][j][k] $和$ s[i][j][k] $ 表示从$ (1,1) $ 到 $ (i,j) $ 这个矩形中比k大的数的个数和这些 ...
- 【BZOJ】2657: [Zjoi2012]旅游(journey)(树的直径)
题目 传送门:QWQ 分析 在任意两个不相邻的点连一条线,求这条线能穿过几个三角形. 建图比较讲究(详见代码) 求树的直径. 代码 #include <bits/stdc++.h> usi ...
- 阶段性总结(PHP-Array函数)
PHP 5 Array 函数 PHP Array 简介 PHP Array 函数允许您访问并操作数组. 支持简单的数组和多维数组. 详情见下表: 函数 描述 array() 创建数组. array_c ...
- java之IO整理(中)
一:打印流/*System.out.println()重定向输出*/ /*public static void main(String[] args) { System.out.println(&qu ...
- 6、数据类型四:sets
集合的基本特征:无序,唯一. 一个redis集合最多能存放232-1个集合元素.其强大之处在于它支持集合的“交.差.并”运算,而且能够快速的判断某个给定值是否在集合中. 1.基本命令: SADD ke ...