【模板】负环
题目描述
给定一个 n n n 个点的有向图,请求出图中是否存在从顶点 1 1 1 出发能到达的负环。
负环的定义是:一条边权之和为负数的回路。
输入格式
本题单测试点有多组测试数据。
输入的第一行是一个整数 T T T,表示测试数据的组数。对于每组数据的格式如下:
第一行有两个整数,分别表示图的点数 n n n 和接下来给出边信息的条数 m m m。
接下来 m m m 行,每行三个整数 u , v , w u, v, w u,v,w。
- 若 w ≥ 0 w \geq 0 w≥0,则表示存在一条从 u u u 至 v v v 边权为 w w w 的边,还存在一条从 v v v 至 u u u 边权为 w w w 的边。
- 若 w < 0 w < 0 w<0,则只表示存在一条从 u u u 至 v v v 边权为 w w w 的边。
输出格式
对于每组数据,输出一行一个字符串,若所求负环存在,则输出 YES
,否则输出 NO
。
样例 #1
样例输入 #1
2
3 4
1 2 2
1 3 4
2 3 1
3 1 -3
3 3
1 2 3
2 3 4
3 1 -8
样例输出 #1
NO
YES
提示
数据规模与约定
对于全部的测试点,保证:
- 1 ≤ n ≤ 2 × 1 0 3 1 \leq n \leq 2 \times 10^3 1≤n≤2×103, 1 ≤ m ≤ 3 × 1 0 3 1 \leq m \leq 3 \times 10^3 1≤m≤3×103。
- 1 ≤ u , v ≤ n 1 \leq u, v \leq n 1≤u,v≤n, − 1 0 4 ≤ w ≤ 1 0 4 -10^4 \leq w \leq 10^4 −104≤w≤104。
- 1 ≤ T ≤ 10 1 \leq T \leq 10 1≤T≤10。
提示
请注意, m m m 不是图的边数。
不详细解释了,需要的可以看看别的题解。
主要思路就是用spfa判断入队次数是否>=n,如果是则说明有负环,这一点可以由spfa的松弛算法原理推导来。
注意一定要判入队次数而不是松弛次数,我看几乎现有每一篇题解判的都是松弛次数,可以试试这个hack。
hack原理很简单:如果存在重边导致了多次松弛,那么对松弛次数的判断就会产生影响。解决方式就是判入队次数,虽然略慢,但是更稳。
update[2020.7.26]:在写差分约束的时候想用spfa判无解,然后经过一系列的思考就有了下面这组新的hack数据:
input:
1
4 6
1 2 -3
1 3 -2
1 4 -1
2 3 -6
2 4 -5
3 4 -4
output:
NO
注意这组hack只对用链式前向星(而非vector)存边且判的是松弛次数(而非入队次数)的有效,而且该数据无重边无自环,比discuss里面的那个数据更有说服力。
首先hack原理就是对n号节点进行n-1轮松弛,每轮都有x(x∈[1,n−1]x \in [1,n-1]x∈[1,n−1])次松弛,这样就能产生n^2级别的松弛次数。
但是判入队次数就hack不掉了,每轮的第一次松弛会让n节点入队,但n节点只有在下一轮才会出队;因此本轮的其余所有松弛全部无法导致入队。
#include<cstdio>
#include<cstring>
#include<queue>
#define inf 0x3f3f3f3f
using namespace std;
const int MAXN=2010;
const int MAXM=3010;
int n,m;
int en=-1,eh[MAXN];
struct edge
{
int u,v,w,next;
edge(int U=0,int V=0,int W=0,int N=0):u(U),v(V),w(W),next(N){}
};edge e[MAXM<<1];
inline void add_edge(int u,int v,int w)
{
e[++en]=edge(u,v,w,eh[u]);eh[u]=en;
}
void input()
{
scanf("%d %d",&n,&m);
en=-1;
memset(eh,-1,sizeof(eh));
int u,v,w;
for(int i=1;i<=m;++i)
{
scanf("%d %d %d",&u,&v,&w);
add_edge(u,v,w);
if(w>=0)add_edge(v,u,w);
}
}
int dis[MAXN],cnt[MAXN];
bool vis[MAXN];
queue<int> q;
void spfa()
{
fill(dis+1,dis+n+1,inf);
memset(cnt,0,sizeof(cnt));
memset(vis,0,sizeof(vis));
while(!q.empty())q.pop();
dis[1]=0;vis[1]=1;q.push(1);
int u,v,w;
while(!q.empty())
{
u=q.front();vis[u]=0;q.pop();
for(int i=eh[u];i!=-1;i=e[i].next)
{
v=e[i].v;w=e[i].w;
if(dis[u]+w<dis[v])
{
dis[v]=dis[u]+w;
if(!vis[v])
{
if(++cnt[v]>=n)//注意就是这个位置的判断。一定要保证在判vis之后,即判入队次数;而不是在判vis之前,即判松弛次数!!!
{
printf("YES\n");return;
}
vis[v]=1;q.push(v);
}
}
}
}
printf("NO\n");
}
int main()
{
// freopen("in.txt","r",stdin);
int t;
scanf("%d",&t);
for(int i=1;i<=t;++i)
{
input();
spfa();
}
return 0;
}
update[2020.7.28]:感谢AK新手村dalao提供的判最短路径边数的思路
在fyy2603提供的hack中,提到了极限数据可能爆int的问题,其原因在于要让入队次数达到上界n,则遍历边的总个数最大可达n2n^2n2。
考虑换一种思路,我们知道如果没有负环,从1号点到每个点的最短路径应当是不存在环的;而如果存在环那它只可能是负环,且最短路径长度会在算法过程中无限增大。
因此我们可以判断1号点到i号点的最短路径长度是否<n(即经过的点数<=n,没有任何一个点被重复经过),来更高效地判断是否存在负环。
而这也就完美避免了极限数据爆int的问题。(直接开ll不香吗)
//其他部分没有区别
void spfa()
{
fill(dis+1,dis+n+1,inf);
memset(cnt,0,sizeof(cnt));
memset(vis,0,sizeof(vis));
while(!q.empty())q.pop();
dis[1]=0;vis[1]=1;q.push(1);
int u,v,w;
while(!q.empty())
{
u=q.front();vis[u]=0;q.pop();
for(int i=eh[u];i!=-1;i=e[i].next)
{
v=e[i].v;w=e[i].w;
if(dis[u]+w<dis[v])
{
dis[v]=dis[u]+w;
cnt[v]=cnt[u]+1;//记录最短路径的边数
if(cnt[v]>=n)//最短路径边数>=n,即存在被重复遍历的点,也就是存在负环
{
printf("YES\n");
return;
}
if(!vis[v])
{
vis[v]=1;q.push(v);
}
}
}
}
printf("NO\n");
}