UVALive - 6800 The Mountain of Gold?(Bellman-ford找负权回路,dfs)

题目链接

https://icpcarchive.ecs.baylor.edu/external/68/6800.pdf

bellman-ford照模板打了一段,能够找到负权回路,问题就是判断0点在不在负权回路中了,于是写了个记忆化dfs。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<vector>
using namespace std;

#define MAX 0x3f3f3f3f
#define N 10100
int nodenum, edgenum;  

typedef struct Edge
{
	int u, v;
	int cost;
}Edge;

Edge edge[N];
int dis[N];

bool vis[N], dp[N];
vector<int> G[N];
bool dfs(int n)
{
	if (n == 0) return true;
	if (vis[n]) return dp[n];
	vis[n] = true;

	for (int i = 0; i < G[n].size(); i++)
	{
		if (dfs(G[n][i]))
		{
			return dp[n] = true;
		}
	}
	return dp[n]=false;
}

bool Bellman_Ford()
{
	for (int i = 0; i < nodenum; ++i)
		dis[i] = (i == 0 ? 0 : MAX);
	for (int i = 0; i < nodenum - 1; ++i)
	for (int j = 0; j < edgenum; ++j)
	if (dis[edge[j].v] > dis[edge[j].u] + edge[j].cost)
	{
		dis[edge[j].v] = dis[edge[j].u] + edge[j].cost;
	}
	bool flag = 1;
	for (int i = 0; i < edgenum; ++i)
	if (dis[edge[i].v] > dis[edge[i].u] + edge[i].cost)
	{
		if (dfs(edge[i].u))
		{
			flag = 0;
			break;
		}
	}
	return flag;
}

int main()
{
	int casen;
	cin >> casen;
	for (int cas = 1; cas <= casen;cas++)
	{
		scanf("%d%d", &nodenum, &edgenum);
		memset(dis, 0, sizeof(dis));
		memset(vis, false, sizeof(vis));
		memset(dp, false, sizeof(dp));
		dp[0] = vis[0] = true;
		for (int i = 0; i < N; i++)
			G[i].clear();
		for (int i = 0; i < edgenum; i++)
		{
			scanf("%d%d%d", &edge[i].u, &edge[i].v, &edge[i].cost);
			G[edge[i].u].push_back(edge[i].v);
		}
		int ok=Bellman_Ford();

		if(!ok)printf("Case #%d: possible\n", cas);
		else printf("Case #%d: not possible\n", cas);
	}
}
时间: 2024-10-14 17:47:20

UVALive - 6800 The Mountain of Gold?(Bellman-ford找负权回路,dfs)的相关文章

poj 3259 bellman最短路判断有无负权回路

Wormholes Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 36717   Accepted: 13438 Description While exploring his many farms, Farmer John has discovered a number of amazing wormholes. A wormhole is very peculiar because it is a one-way p

Bellman - Ford 算法解决最短路径问题

Bellman - Ford 算法: 一:基本算法 对于单源最短路径问题,上一篇文章中介绍了 Dijkstra 算法,但是由于 Dijkstra 算法局限于解决非负权的最短路径问题,对于带负权的图就力不从心了,而Bellman - Ford算法可以解决这种问题. Bellman - Ford 算法可以处理路径权值为负数时的单源最短路径问题.设想可以从图中找到一个环路且这个环路中所有路径的权值之和为负.那么通过这个环路,环路中任意两点的最短路径就可以无穷小下去.如果不处理这个负环路,程序就会永远运

Bellman—Ford算法思想

---恢复内容开始--- Bellman—Ford算法能在更普遍的情况下(存在负权边)解决单源点最短路径问题.对于给定的带权(有向或无向)图G=(V,E),其源点为s,加权函数w是边集E的映射.对图G运行Bellman—Ford算法的结果是一个布尔值,表明图中是否存在着一个从源点s可达的负权回路.若存在负权回路,单源点最短路径问题无解:若不存在这样的回路,算法将给出从源点s到图G的任意顶点v的最短路径值d[v] Bellman—Ford算法流程 分为三个阶段: (1)初始化:将除源点外的所有顶点

ACM/ICPC 之 最短路径-Bellman Ford范例(POJ1556-POJ2240)

两道Bellman Ford解最短路的范例,Bellman Ford只是一种最短路的方法,两道都可以用dijkstra, SPFA做. Bellman Ford解法是将每条边遍历一次,遍历一次所有边可以求得一点到任意一点经过一条边的最短路,遍历两次可以求得一点到任意一点经过两条边的最短路...如 此反复,当遍历m次所有边后,则可以求得一点到任意一点经过m条边后的最短路(有点类似离散数学中邻接矩阵的连通性判定) POJ1556-The Doors 初学就先看POJ2240吧 题意:求从(0,5)到

POJ 1860 Currency Exchange (Bellman ford)

Currency Exchange Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 22405   Accepted: 8095 Description Several currency exchange points are working in our city. Let us suppose that each point specializes in two particular currencies and pe

ACM/ICPC 之 Bellman Ford练习题(ZOJ1791(POJ1613))

这道题稍复杂一些,需要掌握字符串输入的处理+限制了可以行走的时间. ZOJ1791(POJ1613)-Cave Raider //限制行走时间的最短路 //POJ1613-ZOJ1791 //Time:16Ms Memory:324K #include<iostream> #include<cstring> #include<cstdio> #include<algorithm> using namespace std; #define MAX 505 #d

poj 3259 Wormholes (BELLman—FOrd算法)(邻接矩阵表示)

http://poj.org/problem?id=3259 之前一开始 ,没做出来,搁置了好几天才看见bug所在.所以今天a掉了 ,把代码贴出来,这个使用邻接矩阵表示的 ,下一篇用邻接表可以提高效率. #include<iostream> #include<queue> #include<stdio.h> #include<string.h> using namespace std; const int INF=600; int G[INF][INF];

Bellman ford 最短路径算法

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" /> 下表记录S到每一个节点的距离: 第一次迭代, S->A = 4 ,由于S->A眼下为INF.因此更新MIN(S->A)为4 S->B = 6.由于S->B眼下为INF.因此更新MIN(S->B)为6 S->C

最短路径——Bellman Ford算法(C++)

源代码: #include<cstdio>#include<cstring>int m(1),n,k,i[1001],x[1001],y[1001],f[1001];int main(){ scanf("%d%d",&n,&k); for (int a=1;a<=n;a++) for (int b=1;b<=n;b++) { scanf("%d",&i[m]); if (i[m]!=-1) { x[m]=a