【模板】负环 问题题解(spfa和bellman解决)

2024-02-18 02:12

本文主要是介绍【模板】负环 问题题解(spfa和bellman解决),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

P3385 【模板】负环

题目描述

给定一个 n 个点的有向图,请求出图中是否存在从顶点 11 出发能到达的负环。

负环的定义是:一条边权之和为负数的回路。

输入格式

本题单测试点有多组测试数据

输入的第一行是一个整数 T,表示测试数据的组数。对于每组数据的格式如下:

第一行有两个整数,分别表示图的点数 n 和接下来给出边信息的条数 m。

接下来 m 行,每行三个整数 u,v,w。

  • 若 w≥0,则表示存在一条从 u 至 v 边权为 w 的边,还存在一条从 v 至 u 边权为 w 的边。
  • 若 w<0,则只表示存在一条从 u 至 v 边权为 w 的边。

输出格式

对于每组数据,输出一行一个字符串,若所求负环存在,则输出 YES,否则输出 NO

输入输出样例

输入 #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×103,1≤m≤3×103。
  • 1≤u,v≤n,−−104≤w≤104。
  • 1≤T≤10。

提示

请注意,m 不是图的边数。

 

 

 


 

 对于存在负环的问题,我们通常使用bellman或者spfa算法(bellman的队列优化)判断有无负环

 

 bellman解决思路:

Bellman-Ford 算法通过不断迭代计算最短路,每轮迭代至少有一个结点得到了最短路。所以,若图中没有负环,则最多经过 n−1 轮迭代后算法结束。若第 n 轮迭代仍有结点的最短路能被更新,则图中有负环。复杂度为O(nm)。

 bellman代码

#include<bits/stdc++.h>
using namespace std;
#define N 2000005
int t, n, m, ans=0, num, sum;
bool vis[N];
int  dis[N];
struct node {int head, to, tail;
}f[N];                    //结构体数组存边void add(int a, int b, int c)  //这里需要判断边的正负,正负情况不同
{if (c >= 0){f[++ans] = { a,b,c };      //注意这里没有使用链式前向星或者邻接表,就是单纯的存边f[++ans] = { b,a,c };}if (c < 0){f[++ans] = { a,b,c };}
}bool bellman()             //bellman核心代码段
{dis[1] = 0;for (int i = 2; i <= n; i++)    //初始化dis数组为无穷大dis[i] = 0x3f3f3f;for (int i = 1; i <= n - 1; i++) {  //只最多需要n-1次,这里可以优化for (int j = 1; j <= ans; j++) {int w = f[j].head, v = f[j].to;   //这里一定要加上对dis的特判,因为测试点中if (dis[v] > dis[w] + f[j].tail&&dis[w]!=0x3f3f3f) {   //有非联通图dis[v] = dis[w] + f[j].tail;}}}for (int k = 1; k <= ans; k++) {         //第n次更新,如果dis还在更新边,存在负环if (dis[f[k].head] == 0x3f3f3f || dis[f[k].to] == 0x3f3f3f)continue;if (dis[f[k].to] > dis[f[k].head] + f[k].tail)return true;}return false;
}
int main()
{cin >> t;while (t--) {memset(f, 0, sizeof(f));          //这里有多组数据,要清空cin >> n >> m;for (int i = 1; i <= m; i++) {int a, b, c;cin >> a >> b >> c;add(a, b, c);}if (bellman())              //判断条件cout << "YES" << endl;elsecout << "NO" << endl;}return 0;
}

 

 

 


spfa解决思路:

spfa和bfs过程很相似,在spfa过程中一个点可以多次入队,更新距离,但是到某个点所更新的边最多有n-1个,即最坏的情况是一条直线,但如果更新的边的数量>=n,那么就说明存在负环,我们可以用一个数组ans,来存储到某个点所需要更新的边的数量,ans[1]=0,每次松弛时,我们使ans[y]=ans[x]+1;

spfa代码

#include<bits/stdc++.h>
using namespace std;
#define N 100005
int head[N], dis[N], ans[N];  //前面都是老伙计了,就不多介绍了
int t, n, m, cnt=1 , sum;
bool vis[N];
struct node {int to, tail, nx;
}f[N];void add(int a, int b, int c)     //链式前向星存边
{f[cnt].to = b;f[cnt].tail = c;f[cnt].nx = head[a];head[a] = cnt++;
}bool spfa()
{memset(dis, 0x3f, sizeof(dis));    //每次调用都有重新初始化memset(vis, 0, sizeof(vis));memset(ans, 0, sizeof(ans));queue<int>q;               //在函数体内定义,相当于每次进行清空q.push(1);dis[1] = 0;vis[1] = 1;       //这里出队后要回复vis,因为可能多次入队while (!q.empty()) {int x = q.front();q.pop();vis[x] = 0;for (int i = head[x]; i; i = f[i].nx) {  //这里应该比较熟悉吧,就是松弛操作int h = f[i].to;if (dis[h] > dis[x] + f[i].tail) {dis[h] = dis[x] + f[i].tail;ans[h] = ans[x] + 1;        //关键步骤,记录更新边的次数if (ans[h] >= n)         //存在负环return true;if (!vis[h]) {           //不在队列中,入队q.push(h);vis[h] = 1;}}}}return false;
}
int main()
{cin >> t;while (t--) {cin >> n >> m;cnt = 1;      //注意对于不同操作的链式前向星,cnt的初始化值不同,我这里是1memset(head, 0, sizeof(head));  //清空操作,多组数据for (int i = 1; i <= m; i++) {int a, b, c;cin >> a >> b >> c;add(a, b, c);if (c >= 0)         //特殊条件add(b, a, c);}if (spfa())cout << "YES" << endl;elsecout << "NO" << endl;}return 0;
}

 

 以上两种算法可以判断是否存在负环,根据需要合理的选择适合的算法。

这篇关于【模板】负环 问题题解(spfa和bellman解决)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/719724

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

如何解决线上平台抽佣高 线下门店客流少的痛点!

目前,许多传统零售店铺正遭遇客源下降的难题。尽管广告推广能带来一定的客流,但其费用昂贵。鉴于此,众多零售商纷纷选择加入像美团、饿了么和抖音这样的大型在线平台,但这些平台的高佣金率导致了利润的大幅缩水。在这样的市场环境下,商家之间的合作网络逐渐成为一种有效的解决方案,通过资源和客户基础的共享,实现共同的利益增长。 以最近在上海兴起的一个跨行业合作平台为例,该平台融合了环保消费积分系统,在短

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

poj 1511 Invitation Cards(spfa最短路)

题意是给你点与点之间的距离,求来回到点1的最短路中的边权和。 因为边很大,不能用原来的dijkstra什么的,所以用spfa来做。并且注意要用long long int 来存储。 稍微改了一下学长的模板。 stack stl 实现代码: #include<stdio.h>#include<stack>using namespace std;const int M

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k