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个不同的客户 ...
随机推荐
- python TypeError: unsupported operand type(s) for +: 'int' and 'str' [closed]
TypeError: unsupported operand type(s) for +: 'int' and 'str' [closed] sql="insert into auto_tr ...
- Java的八种基本数据类型及其包装类
Java有八种基本数据类型,所谓基本类型就是说存储时仅存在栈中,那么与之相对就是引用类型,引用类型既存在栈里又存在堆里,栈内存放堆内地址. 八种基本类型分别为byte short int long f ...
- 【JS】手机屏幕旋转判断
function readDeviceOrientation() { if (Math.abs(window.orientation) === 90) { // Landscape alert('横屏 ...
- java代码--------打印三角形
总结:这里主要是for循环里的j<=i而不死j<=i-1;.还有先打印“*” 再打印空格.换行.理解.请用脑子啊 package com.sads; public class Dds { ...
- [Java.Web] Servlet 的一些细节
本文来自 传智播客视频PPT 1. 由于客户端是通过 URL 地址访问 web 服务器中的资源,所以 Servlet 程序若想被外界访问,必须把 servlet 程序映射到一个 URL 地址上,这个工 ...
- mysql 实现row_number,获取上一条,下一条
代码思路如下: select *,(@row_num:=@row_num+1) as row_no from stc_output,(select(@row_num:=0)) b; select * ...
- 【洛谷】P1052 过河(状压dp)
题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数 ...
- Linux: su sudo sudoer
日常操作中为了避免一些误操作,更加安全的管理系统,通常使用的用户身份都为普通用户,而非root.当需要执行一些管理员命令操作时,再切换成root用户身份去执行. 普通用户切换到root用户的方式有:s ...
- C# using语句的使用
使用时注意事项 ①using只能用于实现了IDisposable接口的类型,禁止为不支持IDisposable接口的类型使用using语句,否则会出现编译错误:②using语句适用于清理单个非托管资源 ...
- C# 多线程窗体的创建
从目前已经在项目中工作将近一个月来的情况来看,凡是费时的操作,基本上都要用到多线程的等待窗体.进度提示窗体等实时显示动态的进度信息.而如果直接在主线程的窗体上实时更新信息,就会造成更新太快或者太慢而出 ...