程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> HDU 1317 XYZZY Bellman-Ford求最長路 判斷正環

HDU 1317 XYZZY Bellman-Ford求最長路 判斷正環

編輯:C++入門知識

題意:給你n個房間 開始有能量值100 判斷能否從1到第n個房間

每到一個房間可以獲得能量x(可能小於0) 每到一個房間總能量必須大於0 每個房間可以重復到達

思路:求一個從1到n的最長路 不過可能有正環 沒有正環 直接求最長路 如果有正環 判斷環中的點是否可以到達n

具體用Bellman-Ford算法 雖然復雜度是(n*m)這題應該可以了 如果迭代n-1次之後還能松弛 說明有正環 然後用floyd判斷是否可達

#include 
#include 
#include 
#include 
using namespace std;
const int maxn = 110;
struct edge
{
	int u, v, w;
};

vector  G;
int dis[maxn];
bool vis[maxn];
int n, m;
int a[maxn][maxn];
void floyd()
{
	for(int k = 1; k <= n; k++)
		for(int i = 1; i <= n; i++)
			for(int j = 1; j <= n; j++)
				a[i][j] = a[i][j] || (a[i][k] && a[k][j]);
		
}
bool Bellman_Ford()
{
	for(int i = 1; i <= n; i++)
		dis[i] = -999999999;
	dis[1] = 100;
	for(int i = 1; i < n; i++)
	{
		for(int j = 0; j < G.size(); j++)
		{
			edge e = G[j];
			if(dis[e.v] < dis[e.u] + e.w && dis[e.u] + e.w > 0)
				dis[e.v] = dis[e.u] + e.w;	
		}
	}
	//printf("%d\n", dis[n]);
	if(dis[n] > 0)
		return true;
	for(int i = 0; i < G.size(); i++)
	{
		edge e = G[i];
		if(dis[e.v] < dis[e.u] + e.w && dis[e.u] + e.w > 0)
		{
			//puts("sss");
			dis[e.v] = dis[e.u] + e.w;	
			if(a[e.v][n])
				return true;
		}
	}
	return false;
}
int main()
{
	
	while(scanf("%d", &n) && n != -1)
	{
		//for(int i = 0; i <= n; i++)
		G.clear();
		memset(a, 0, sizeof(a));
		for(int i = 1; i <= n; i++)
		{
			int t, v, w;
			scanf("%d %d", &w, &t);
			while(t--)
			{
				scanf("%d", &v);
				G.push_back((edge){i, v, w});
				//G.push_back((edge){v, i, w});
				a[i][v] = 1;
				//a[v][i] = 1;
				//G[v].push_back((edge){u, w});
			}
		}
		floyd();
		if(Bellman_Ford())
			puts("winnable");
		else
			puts("hopeless");
	}
	return 0;
}


  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved