[USACO11JAN]道路和飞机Roads and Planes
时间:2019-09-20
本文章向大家介绍[USACO11JAN]道路和飞机Roads and Planes,主要包括[USACO11JAN]道路和飞机Roads and Planes使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
洛咕
题意:有n个点,m条无向边(权值非负),k条有向边(权值可能为负数),给定起点s,求出s点到每个点的最短路径长度.\(n<=25000,m,k<=50000\)
分析:
方法一:既然有负权边,那就跑SPFA咯,但是普通队列SPFA只有88分,要使用到\(SLF\)的优化策略,即把普通队列改为双端队列,每次要入队时,将dis值与当前队头节点的dis值比较,如果小,则从队头入队,否则从队尾入队.这样就能水过去了.
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define ll long long
using namespace std;
inline int read(){
int x=0,o=1;char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
if(ch=='-')o=-1,ch=getchar();
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x*o;
}
const int N=50005;
const int M=200005;
int n,r,p,s;
int dis[N],visit[N];
int tot,head[N],nxt[M],to[M],w[M];
inline void add(int a,int b,int c){
nxt[++tot]=head[a];head[a]=tot;
to[tot]=b;w[tot]=c;
}
deque<int>q;
inline void spfa(){
for(int i=1;i<=n;++i)dis[i]=1e9;
q.push_back(s);visit[s]=1;dis[s]=0;
while(q.size()){
int u=q.front();q.pop_front();visit[u]=0;
for(int i=head[u];i;i=nxt[i]){
int v=to[i];
if(dis[v]>dis[u]+w[i]){
dis[v]=dis[u]+w[i];
if(!visit[v]){
if(dis[v]<=dis[q.front()])q.push_front(v);//队头入队
else q.push_back(v);//队尾入队
visit[v]=1;
}
}
}
}
}
int main(){
n=read();r=read();p=read();s=read();
for(int i=1;i<=r;++i){
int a=read(),b=read(),c=read();
add(a,b,c);add(b,a,c);
}
for(int i=1;i<=p;++i){
int a=read(),b=read(),c=read();
add(a,b,c);
}
spfa();
for(int i=1;i<=n;++i){
if(dis[i]==1e9)puts("NO PATH");
else printf("%d\n",dis[i]);
}
return 0;
}
方法二:先只把双向边加入图中,则会得到若干个联通块,我们把这些联通块看成一个个"点",那么加入有向边之后就是个\(DAG\),对于\(DAG\),我们不用管权值的正负,只要拓扑排序跑就好了.
具体说来,先只把双向边加入图中,跑DFS预处理出每个点属于哪个联通块\(belong[]\).然后加入有向边,并统计每个联通块的入度\(deg[]\).把入度为0的联通块和起点S所在的联通块丢进队列里面,开始拓扑排序,对于队列里面的每个节点(联通块),把联通块所包含的点全都丢进堆里面跑\(dij\).
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define ll long long
using namespace std;
inline int read(){
int x=0,o=1;char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
if(ch=='-')o=-1,ch=getchar();
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x*o;
}
const int N=25005;
const int M=200005;
int n,m,k,s,num;
int belong[N],deg[N],dis[N],visit[N];
int tot,head[N],nxt[M],to[M],w[M];
inline void add(int a,int b,int c){
nxt[++tot]=head[a];head[a]=tot;
to[tot]=b;w[tot]=c;
}
vector<int>Q[N];
inline void dfs(int u){
belong[u]=num;Q[num].push_back(u);
for(int i=head[u];i;i=nxt[i]){
int v=to[i];if(belong[v])continue;
dfs(v);
}
}
queue<int>q;
struct ppx{
int id,val;
bool operator <(const ppx &x)const{
return val>x.val;
}
}temp;
priority_queue<ppx>qq;
inline void topsort(){
q.push(belong[s]);//起点所在的联通块丢进队列
for(int i=1;i<=num;++i)if(!deg[i])q.push(i);//入度为0,丢进队列
memset(dis,0x3f,sizeof(dis));dis[s]=0;
while(q.size()){
int u=q.front();q.pop();
for(int j=0;j<Q[u].size();++j){
temp.id=Q[u][j];temp.val=dis[Q[u][j]];
qq.push(temp);
}//把联通块内的所有点丢进堆里
while(qq.size()){//dij模板
temp=qq.top();qq.pop();
int uu=temp.id;
if(visit[uu])continue;visit[uu]=1;
for(int i=head[uu];i;i=nxt[i]){
int vv=to[i];
if(dis[vv]>dis[uu]+w[i]){
dis[vv]=dis[uu]+w[i];
if(belong[uu]==belong[vv]){
temp.id=vv;temp.val=dis[vv];
qq.push(temp);
}
}
if(belong[uu]!=belong[vv]){//维护拓扑排序
--deg[belong[vv]];
if(!deg[belong[vv]])q.push(belong[vv]);
}
}
}
}
}
int main(){
n=read();m=read();k=read();s=read();
for(int i=1;i<=m;++i){
int a=read(),b=read(),c=read();
add(a,b,c);add(b,a,c);//先只建无向边
}
for(int i=1;i<=n;++i)if(!belong[i])++num,dfs(i);//预处理联通块
for(int i=1;i<=k;++i){
int a=read(),b=read(),c=read();
add(a,b,c);++deg[belong[b]];//加入有向边,并统计每个联通块的入度
}
topsort();//拓扑排序
for(int i=1;i<=n;++i){
if(dis[i]>1e9)puts("NO PATH");//刚开始写dis[i]==1e9,只有30分,奇怪!!!
else printf("%d\n",dis[i]);
}
return 0;
}
原文地址:https://www.cnblogs.com/PPXppx/p/11557697.html
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 浅谈优化Django ORM中的性能问题
- PHP chr()函数讲解
- iOS音视频接入 - TRTC常见问题
- PHP常见的几种攻击方式实例小结
- PHP strripos函数用法总结
- Laravel5.0+框架邮件发送功能实现方法图文与实例详解
- Ajax+PHP实现的删除数据功能示例
- tensorflow下的图片标准化函数per_image_standardization用法
- 浅析Python面向对象编程
- Python单元测试及unittest框架用法实例解析
- Tensorflow中批量读取数据的案列分析及TFRecord文件的打包与读取
- YII框架实现自定义第三方扩展操作示例
- 在Tensorflow中实现leakyRelu操作详解(高效)
- Django def clean()函数对表单中的数据进行验证操作
- Python3爬虫中Splash的知识总结