判断负环: spfa

80 阅读1分钟

dist[]数组维护最短距离

cnt[]数组维护边的最小个数

每次更新一下

dist[x]=dist[t]+w[i]
cnt[x]=cnt[t]+1

cnt[x]=cnt[t]+1的意思是:1到x的最小边数是1到t的最小边数,再加上一条边:

image.png

如果说cnt[]>=n,那说明至少经过了n+1个点(这样才会至少经过n条边),一共至少n个点,却经过了n+1个点,说明有一个点至少经过了两次,也就是负环了。

code

注意,题目要求求负环,而不是让求从1开始的负环,也就是从1开始不一定能到达负环。因为我们刚开始就不能只把1号节点加入队列·1,而是应该把所有的节点全部加入队列中,这样只有存在负环就一定能找到。

#include<bits/stdc++.h>
using namespace std;
const int N = 1e5 + 10;
int h[N], e[N], w[N], ne[N], idx;
int dist[N],cnt[N];
bool st[N];
int n, m;

void add(int a, int b, int c)
{
    e[idx] = b;
    w[idx] = c;
    ne[idx] = h[a];
    h[a] = idx++;
}
int spfa()
{
    queue<int>q;
    for(int i=1;i<=n;i++)
    { st[i]=true;
        q.push(i);
       
    }
    while (q.size())
    {
        int t = q.front();
        q.pop();

        st[t] = false;

        for (int i = h[t]; i != -1; i = ne[i])
        {
            int j = e[i];
            if (dist[j] > dist[t] + w[i])
            {
                dist[j] = dist[t] + w[i];
                cnt[j]=cnt[t]+1;
                
                if(cnt[j]>=n)return true;
                if (!st[j])
                {
                    q.push(j);
                    st[j] = true;
                }
            }
        }
    }
    return false;
}


int main()
{
    memset(h, -1, sizeof h);
    cin >> n >> m;
    while (m--)
    {
        int a, b, c; cin >> a >> b >> c;
        add(a, b, c);
    }
    if(spfa())cout<<"Yes"<<endl;
    else cout<<"No"<<endl;
    return 0;
}