vijos    1053    Easy sssp

方法:用spfa判断是否存在负环

描述

输入数据给出一个有N(2 <= N <= 1,000)个节点,M(M <= 100,000)条边的带权有向图. 
要求你写一个程序, 判断这个有向图中是否存在负权回路. 如果从一个点沿着某条路径出发, 又回到了自己, 而且所经过的边上的权和小于0, 就说这条路是一个负权回路.
如果存在负权回路, 只输出一行-1;
如果不存在负权回路, 再求出一个点S(1 <= S <= N)到每个点的最短路的长度. 约定: S到S的距离为0, 如果S与这个点不连通, 则输出NoPath.

格式

输入格式

第一行: 点数N(2 <= N <= 1,000), 边数M(M <= 100,000), 源点S(1 <= S <= N);
以下M行, 每行三个整数a, b, c表示点a, b(1 <= a, b <= N)之间连有一条边, 权值为c(-1,000,000 <= c <= 1,000,000)

输出格式

如果存在负权环, 只输出一行-1, 否则按以下格式输出
共N行, 第i行描述S点到点i的最短路: 
如果S与i不连通, 输出NoPath;
如果i = S, 输出0;
其他情况输出S到i的最短路的长度.

样例1

样例输入1

6 8 1
1 3 4
1 2 6
3 4 -7
6 4 2
2 4 5
3 6 3
4 5 1
3 5 4

样例输出1

0
6
4
-3
-2
7

限制

Test5 5秒
其余 1秒

提示

做这道题时, 你不必为超时担心, 不必为不会算法担心, 但是如此“简单”的题目, 你究竟能ac么?

思路:哈哈,用spfa判断是否存在负环。

注意:1.要开long long 不然wa一个点

   2.写读入优化,至少不用cin,cout

   3.本题我在跑是否存在负环时,跑了两遍spfa,如果存在负环就不跑第二遍求最短路的那一遍了,第二遍spfa求最短路。

  4.在判断负环是查询一个点被访问过几次,若被访问过n次,则说明存在负环。(他的思路和spfa一样!板子差不多)

唉,直接不容易啊!!

代码:

#include<queue>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 100010
#define maxn 999999999
using namespace std;
int read()
{
    ,f=;
    char ch=getchar();
    ')
    {
        ;
        ch=getchar();
    }
    ')
    {
        x=x*+ch-';
        ch=getchar();
    }
    return x*f;
}
struct Edge
{
    int to,ds,next;
}edge[N];
int n,m,s,x,y,z,head[N],tot,sum[N];
long long dis[N];
bool vis[N];
int add(int from,int to,int dis)
{
    tot++;
    edge[tot].ds=dis;
    edge[tot].to=to;
    edge[tot].next=head[from];
    head[from]=tot;
}
int spfa1(int s)//
{
    memset(dis,0x3f,sizeof(dis));
    memset(vis,false,sizeof(vis));
    queue<int>q;
    dis[s]=,vis[s]=true;
    q.push(s);
    while(!q.empty())
    {
        int x=q.front();
        q.pop();vis[x]=false;
        for(int i=head[x];i;i=edge[i].next)
        {
            if(dis[x]+edge[i].ds<dis[edge[i].to])
            {
                dis[edge[i].to]=dis[x]+edge[i].ds;
                sum[edge[i].to]++;
                q.push(edge[i].to);
                if(sum[edge[i].to]>n)
                  ;
            }
        }
    }
    ;
}
void spfa2(int s)//
{
    memset(dis,0x3f,sizeof(dis));
    memset(vis,false,sizeof(vis));
    queue<int>q;
    dis[s]=,vis[s]=true;
    q.push(s);
    while(!q.empty())
    {
        int x=q.front();
        q.pop();vis[x]=false;
        for(int i=head[x];i;i=edge[i].next)
        {
            if(dis[x]+edge[i].ds<dis[edge[i].to])
            {
                dis[edge[i].to]=dis[x]+edge[i].ds;
                if(!vis[edge[i].to])
                    q.push(edge[i].to),vis[edge[i].to]=true;
            }
        }
    }
}
int main()
{
    n=read(),m=read(),s=read();
    ;i<=m;i++)
    {
        x=read(),y=read(),z=read();
        add(x,y,z);
    }
    ;i<=n;i++)
    {
        )
        {
            int ans=spfa1(i);
            )
            {
                printf("-1");
                ;
             }
        }
    }
    spfa2(s);
    ;i<=n;i++)
    {
        if(dis[i]==0x3f3f3f3f3f3f3f3fll) printf("NoPath\n");
        else printf("%lld\n",dis[i]);
    }
    ;
}

最新文章

  1. gulp 配置自动化前端开发
  2. Codeforces 549B. Looksery Party[构造]
  3. MS sql server 基础知识回顾(二)-表连接和子查询
  4. Linux GDB常用命令一栏
  5. Actipro Ribbon For WPF 界面控件免费下载地址
  6. Provider Communication with Apple Push Notification Service
  7. Silicon Labs电容式触摸感应按键技术原理及应用
  8. UNIX网络编程 12 15共享内存区
  9. Win10系统下安装Oracle服务器和Oracle客户端
  10. K:常见的正则表达式
  11. RxSwift(一)
  12. 使用COM打开Excel文档注意事项
  13. 探究Java中的锁
  14. mui---获取上一级窗口
  15. hdu3336 Count the string 扩展KMP
  16. 小程序/js监听输入框验证金额
  17. es 5 数组reduce方法记忆
  18. C# toolstrip 上添加DateTimePicker Control控件
  19. 使用CMake编译跨平台静态库
  20. IOS控件大全及控件大小

热门文章

  1. Bootstrap 网格系统(Grid System)实例5
  2. shell脚本,怎么实现每次新开一个shell都输出一个提示语?
  3. Linux基础学习系列目录导航
  4. lnmp一键安装包 虚拟主机问题
  5. IE10无法识别setPrototypeOf属性问题
  6. Luogu 2627 修建草坪 (动态规划Dp + 单调队列优化)
  7. redis安装与安全设置
  8. Python Flask+Mysql练习题
  9. 五、docker配置镜像加速器之阿里云
  10. linux下连接到远程主机,用图像界面(想在远程服务器上用cmake)