【BZOJ】【2200】【USACO 2011 Jan】道路和航线
2024-09-20 08:53:57
做了一天……
TLE:数组开小了-_-#道路是有50000的,双向要乘二。(我特么怎么想的就以为是树了……)
WA:一些大点都WA了,小点都过了。好纠结……
AC了QAQ,不知道为什么,在并查集合并的时候写成fa[x]=y就会WA,写成fa[y]=x就AC……这不是一样的吗?
(虽然说是fa[y]=x是把出边到达的节点的fa都置为x,好像更符合人的思维= =)
Update:由于是宽搜,所以是有“祖先后代“关系的,如果写成fa[x]=y就相当于把所有的祖先后代关系反过来……当然原本是同一系的关系就会变得分裂开了!
/**************************************************************
Problem: 2200
User: Tunix
Language: C++
Result: Accepted
Time:688 ms
Memory:5432 kb
****************************************************************/ //BZOJ 2200
#include<queue>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
void read(int &v){
v=; int sign=; char ch=getchar();
while(ch<''||ch>''){ if (ch=='-') sign=-; ch=getchar();}
while(ch>=''&&ch<=''){ v=v*+ch-''; ch=getchar();}
v*=sign;
}
/******************tamplate*********************/
const int N=,INF=1e9;
int to[N<<],next[N<<],head[N],len[N<<],cnt;
inline void add(int x,int y,int z){
to[++cnt]=y; next[cnt]=head[x]; head[x]=cnt; len[cnt]=z;
to[++cnt]=x; next[cnt]=head[y]; head[y]=cnt; len[cnt]=z;
}
int t[N],ne[N],h[N],l[N],tot;
inline void ins(int x,int y,int z){
t[++tot]=y; ne[tot]=h[x]; h[x]=tot; l[tot]=z;
}
/*******************edge************************/
int in[N],n,R,P,S;
struct node{
int dist,num;
bool operator < (const node &now)const{
return dist>now.dist;
}
};
int dist[N];
priority_queue<node>Q;
bool done[N];
queue<int>tp;//拓扑
int fa[N];
int find(int x){ return fa[x]==x ? x : (fa[x]=find(fa[x]));} vector<int>have[N];
void dijkstra(int x){
int temp=;
rep(i,have[x].size()){
temp=have[x][i];
Q.push((node){dist[temp],temp});
}
while(!Q.empty()){
int now=Q.top().num; Q.pop();
if (done[now]) continue;
done[now]=;
for(int i=head[now];i;i=next[i]){
if (dist[to[i]]>dist[now]+len[i]){
dist[to[i]]=dist[now]+len[i];
Q.push((node){dist[to[i]],to[i]});
}
}
for(int i=h[now];i;i=ne[i]){
int set=find(t[i]);
if (dist[t[i]]>dist[now]+l[i]){
dist[t[i]]=dist[now]+l[i];
have[set].pb(t[i]);
}
in[set]--;
if (in[set]==) tp.push(set);
}
}
}
/******************dijkstra*********************/
bool vis[N];
void bfs(){
queue<int>q;
vis[S]=; q.push(S);
while(!q.empty()){
int x=q.front();
q.pop();
for(int i=head[x];i;i=next[i])
if (!vis[to[i]]) vis[to[i]]=,q.push(to[i]);
for(int i=h[x];i;i=ne[i]){
++in[find(t[i])];
if (!vis[t[i]]) vis[t[i]]=, q.push(t[i]);;
}
}
}
void solve(){
F(i,,n) fa[i]=i;
F(i,,n){
int x=find(i),y;
for(int j=head[i];j;j=next[j]){
y=find(to[j]);
if (x!=y) fa[y]=x;//就是这里!!!想不通啊想不通
}
}
bfs();
//ready
F(i,,n) { dist[i]=INF;done[i]=;}
dist[S]=;
tp.push(find(S));
have[find(S)].pb(S);
//end
while(!tp.empty()){
dijkstra(tp.front());
tp.pop();
}
F(i,,n)
if (dist[i]!=INF) printf("%d\n",dist[i]);
else printf("NO PATH\n");
}
int main(){
read(n); read(R); read(P); read(S);
int x,y,z;
F(i,,R){
read(x); read(y); read(z);
add(x,y,z);
}
F(i,,P){
read(x); read(y); read(z);
ins(x,y,z);
}
solve();
return ;
}
最新文章
- 【Oracle】oracle10g以后利用q-quote特性简化包含单引号后双引号的字符串写法
- 制作图片边框:《CSS3 Border-image》
- Semi-definite programming优化工具
- poj 1012 Joseph (约瑟夫问题)
- 看了一本书,说可以利用Hierarchy Viewer优化布局
- RedHat安装GCC问题-解决依赖问题
- 【Time系列一】datetime的妙用
- VMware虚拟机出现Reason: Failed to lock the file
- Vulkan Tutorial 24 Descriptor pool and sets
- ArcGis SOE(server object extensions)之REST Template初体验
- 记一次DDOS攻击防御实录
- LR socket协议脚本
- Python魔法方法(magic method)细解几个常用魔法方法(下)
- A*搜索算法
- YY:2018互联网创业公司应看清的事情
- YII2中使用控制台命令
- input的type=file触发的相关事件
- Android逆向进阶——让你自由自在脱壳的热身运动(dex篇)
- 2018年秋季学期《C语言程序设计I》教学过程及学期总结
- HTML - SELECT默认选中