SGU103+POJ 1158 最短路/dp
题意:一个无向图,求起点到终点最少时间,限制:每个路口有灯,要灯颜色一样才能过去,灯之有俩种颜色,周期
变化,给定每个灯初态,时间。
思路:开始就想到直接DP,方程dp[k]=dp[i]+distance[i][k]+waittime(i,k),于是天真得BFS敲了这个,
SB啊!有些点松弛了,之后它又被松弛,它便还可以松弛别人啊!否则世界上最短路O(n)的算法就诞生了!
所以,一旦某个点被松弛了,它便可以再入队,啊哈?这不就是SPFA吗!!我去,这次是对SPFA更加理解了,
自己想法竟然遇上经典算法了。其本质就是DP,状态的转移,初期放的错误是:没有所有点都对k松弛,而k已经出队
(对别人松弛过了)。
这题较麻烦的地方是计算等待的时间,细节模拟,我是先算出当前时间,路口的状态,再讨论下接下来等待的
时间。
未1A:
复制差不多代码的时候2个数据没有改,导致错误。打印路劲的时候开始用栈(因为记录是逆序的)。导致爆内存,
以后尽量不要用栈记录!
改进:其实写等待函数可以分俩个函数,每个点当前状态一个,加一个算时间的函数。那样代码就减少很多,
并且也清晰很多。
#include<iostream>
#include<queue>
#include<cstdio>
using namespace std;
const int inf=0x3f3f3f3f;
struct nodes
{
int remain;
char initial;
int tb;
int tp;
};
nodes node[310];
int n,m;int s,l;
int times=0;
int e[50000][3];int head[310];int nume=0;
void inline addedge(int from,int to, int w)
{
e[nume][0]=to;e[nume][1]=head[from];head[from]=nume;
e[nume++][2]=w;
e[nume][0]=from;e[nume][1]=head[to];head[to]=nume;
e[nume++][2]=w;
}
int dp[310];
void readin()
{
cin>>s>>l>>n>>m;
for(int i=1;i<=n;i++)
{
head[i]=-1;
dp[i]=inf;
cin>>node[i].initial>>node[i].remain>>node[i].tb>>node[i].tp;
}
int from,to,w;
for(int i=0;i<m;i++)
{
cin>>from>>to>>w;
addedge(from,to,w);
}
}
char inline change(char s) //变色
{
if(s=='B')return 'P';
return 'B';
}
int gettime(int nowt,int u,int v) //计算等待时间
{
int reu=0,rev=0; //模拟出当前灯还剩余多少时间
int nowu=nowt-node[u].remain; //先减去开始的剩余时间
char cv=node[v].initial,cu=node[u].initial;//cu,cv模拟出当前灯
if(nowu==0) //干好
{
if(node[u].initial=='B'){cu='p';reu=node[u].tp;}
else {cu='B';reu=node[u].tb;}
}
else if(nowu<0) //剩余时间还有
{
reu=node[u].remain-nowt;
}
else
{
int re=nowu%(node[u].tb+node[u].tp);//多余
if(re==0)cu=change(cu); //讨论略繁琐
else if(cu=='B')
{
if(re<node[u].tp)
cu=change(cu);
}
else if(cu=='P')
{
if(re<node[u].tb)
cu=change(cu);
}
if(cu=='P')
{
reu=(node[u].tb+node[u].tp)-re-node[u].tb;
if(node[u].initial=='P')reu+=node[u].tb;
}
else if(cu=='B')
{
reu=(node[u].tb+node[u].tp)-re-node[u].tp;
if(node[u].initial=='B')reu+=node[u].tp;
}
}
int nowv=nowt-node[v].remain;
if(nowv==0)
{
if(node[v].initial=='B'){cv='p';rev=node[v].tp;}
else {cv='B';rev=node[v].tb;}
}
else if(nowv<0)
{
rev=-nowv;
}
else
{
int re=nowv%(node[v].tb+node[v].tp);
if(re==0)cv=change(cv);
else if(cv=='B')
{
if(re<node[v].tp)
cv=change(cv);
}
else if(cv=='P')
{
if(re<node[v].tb)
cv=change(cv);
}
if(cv=='P')
{
rev=(node[v].tb+node[v].tp)-re-node[v].tb;
if(node[v].initial=='P')rev+=node[v].tb;
}
else if(cv=='B')
{
rev=(node[v].tb+node[v].tp)-re-node[v].tp;
if(node[v].initial=='B')rev+=node[v].tp;
}
}
if(cu==cv)return 0;
else
{
if(reu>rev)
return rev;
if(reu<rev)
return reu;
else
{
cu=change(cu);
cv=change(cv);
if(cu=='B')
{
if(node[u].tb<node[v].tp)
return reu+node[u].tb;
else if(node[u].tb>node[v].tp)
return reu+node[v].tp;
else
{
if(node[u].tp<node[v].tb)
return reu+node[u].tb+node[u].tp;
else if(node[u].tp>node[v].tb)
return reu+node[v].tp+node[v].tb;
else
return inf;
}
}
else if(cu=='P')
{
if(node[u].tp<node[v].tb)
return reu+node[u].tp;
else if(node[u].tp>node[v].tb)
return reu+node[v].tb;
else
{
if(node[u].tb<node[v].tp)
return reu+node[u].tp+node[u].tb;
else if(node[u].tb>node[v].tp)
return reu+node[v].tb+node[v].tp;
else
return inf;
}
}
}
}
}
int vis[310];
int fa[310];
int way[310];
queue<int>q;
int main()
{
readin();
q.push(s);
dp[s]=0;
while(!q.empty()) //spfa
{
int cur=q.front();
q.pop();
vis[cur]=0;
for(int i=head[cur];i!=-1;i=e[i][1])
{
int v=e[i][0];
int getv=dp[cur]+e[i][2];
getv+=gettime(dp[cur],cur,v);
if(vis[v]==0&&getv<dp[v])
{
q.push(v);
vis[v]=1;
}
if(getv<inf&&getv<dp[v])
{
dp[v]=getv;
fa[v]=cur; //跟新时候记录父节点,不可记录子节点法,容易想到反例。
}
}
}
if(dp[l]>=inf){cout<<0<<endl;return 0;} //无解
cout<<dp[l]<<endl;
int i=l;
int num=0;
while(i!=s)
{
way[num++]=i;
i=fa[i];
}
way[num]=i;
for(int j=num;j>=0;j--)
{
if(j==0)printf("%d\n",way[j]);
else printf("%d ",way[j]);
}
return 0;
}
SGU103+POJ 1158 最短路/dp的更多相关文章
- POJ 3635 Full Tank? 【分层图/最短路dp】
任意门:http://poj.org/problem?id=3635 Full Tank? Time Limit: 1000MS Memory Limit: 65536K Total Submis ...
- hdu 4568 Hunter 最短路+dp
Hunter Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Subm ...
- BZOJ_1003_[ZJOI2006]物流运输_最短路+dp
BZOJ_1003_[ZJOI2006]物流运输_最短路+dp 题意:http://www.lydsy.com/JudgeOnline/problem.php?id=1003 分析: 这种一段一段的显 ...
- POJ.3624 Charm Bracelet(DP 01背包)
POJ.3624 Charm Bracelet(DP 01背包) 题意分析 裸01背包 代码总览 #include <iostream> #include <cstdio> # ...
- [USACO07NOV]牛继电器Cow Relays (最短路,DP)
题目链接 Solution 非正解 似乎比较蛇啊,先个一个部分分做法,最短路+\(DP\). 在求最短路的堆或者队列中存储元素 \(dis_{i,j}\) 代表 \(i\) 这个节点,走了 \(j\) ...
- P1772 [ZJOI2006]物流运输 最短路+DP
思路:最短路+DP 提交:1次 题解: $f[i]$表示到第$i$天的最小代价,我们可以预先处理出$i,j$两天之间(包括$i,j$)都可通行的最短路的代价记做$s[i][j]$,然后有$f[i]=m ...
- Heavy Transportation POJ 1797 最短路变形
Heavy Transportation POJ 1797 最短路变形 题意 原题链接 题意大体就是说在一个地图上,有n个城市,编号从1 2 3 ... n,m条路,每条路都有相应的承重能力,然后让你 ...
- POJ 2995 Brackets 区间DP
POJ 2995 Brackets 区间DP 题意 大意:给你一个字符串,询问这个字符串满足要求的有多少,()和[]都是一个匹配.需要注意的是这里的匹配规则. 解题思路 区间DP,开始自己没想到是区间 ...
- bzoj1003物流运输 最短路+DP
bzoj1003物流运输 题目描述 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要n天才能运完.货物运输过程中一般要转停好几个码头.物流公司通常会设计一条固定的运输路线,以便对整个运输 ...
随机推荐
- 玩4K必备知识:HDMI1.4、2.0、2.0a、2.0b接口参数对比【扫盲贴】
https://www.4k123.com/thread-55369-1-1.html 前言:玩4K的同学都知道,HDMI接口是视频传输最常用的接口,但是这个接口却有好几个版本HDMI1.4.HDMI ...
- Bootstrap响应式布局(1)
<!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...
- STATIC 和 CLASS
STATIC 和 CLASS 由 王巍 (@ONEVCAT) 发布于 2015/01/28 Swift 中表示 “类型范围作用域” 这一概念有两个不同的关键字,它们分别是 static 和 class ...
- MySQL丨01丨数据库基本概念
以前记录数据可能很少也很简单,比如说老王借了老李半斤肉,这样的数据老李直接就写到墙上就行了. 后来数据多了人们就以表格的方式开始记录,写到一张A4纸上,比如学生的档案,有表头和序号等. 表头里有姓名. ...
- Apache安装错误 APR not found解决方法
在配置Apache的时候,出现错误 原因是缺少一些依赖包,安装这些依赖包就行了 下载依赖包,注意我这里下载的与参考链接上的有些不同,安装上也有不一样 wget http://archive.apach ...
- 各种排序算法(JS实现)
目录: 直接插入排序.希尔排序.简单选择排序.堆排序.冒泡排序.快速排序,归并排序.桶排序.基数排序.多关键字排序.总结 JS测试代码 function genArr(){ let n = Math. ...
- python--操作系统介绍,进程的创建(并发)
一 . 操作系统的作用: 1:隐藏丑陋复杂的硬件接口,提供良好的抽象接口 2:管理.调度进程,并且将多个进程对硬件的竞争变得有序 二 多道技术: 所谓多道程序设计技术,就是指允许多个程序同时进入内存 ...
- 删除mysql主从
在创建数据库主从配置后,若想删除数据库的主从服务可根据以下步骤来删除数据库主从 1.停止slave服务器的主从同步 为了防止主从数据不同步,需要先停止slave上的同步服务, STOP SLAVE ...
- Solr5.0.0 DIH之增量索引
定时索引相关知识 增量更新需要配置个sql(deltaImportQuery.deltaQuery) deltaImportQuery="select * where id='${dih.d ...
- 彻底理解Python中的yield
阅读别人的python源码时碰到了这个yield这个关键字,各种搜索终于搞懂了,在此做一下总结: 通常的for…in…循环中,in后面是一个数组,这个数组就是一个可迭代对象,类似的还有链表,字符串,文 ...