http://poj.org/problem?id=1932
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=1935
题目大意:
看到XYZZY可不要以为是在玩扫雷哦。
给你一张图,初始你在房间1,初始生命值为100,进入每个房间会加上那个房间的生命(可能为负),要你进入房间n,问是否可能。(要求进入每个房间后生命值都大于0)
思路:
1、SPFA求最长路径,如果路径存在(即无环),那么肯定可以。
2、存在负环,不管她,因为如果为负,那就失败了。
3、存在正环,那么说明有无限生命((╯‵□′)╯︵┻━┻ 开挂啊,快举报!),不过无限生命也要求你可达n(我WA了就是因为这个!)
那么如何判断存在环?
用SPFA中,如果一个点入队超过n次,说明存在环。 - -|||
#include<cstdio> #include<cstring> #include<queue> using namespace std; const int INF=-9999999; const int MAXN=111;//注孤 - -||| bool map[MAXN][MAXN]; int val[MAXN],dis[MAXN]; int n; bool SPFA() { for(int i=1;i<=n;i++) dis[i]=INF; bool vis[MAXN]={0}; int cnt[MAXN]={0}; queue<int> q; q.push(1); vis[1]=true; dis[1]=100; cnt[1]=1; while(!q.empty()) { int cur=q.front(); q.pop(); if(cnt[cur] > n) break; vis[cur]=false; for(int i=1;i<=n;i++) { if(map[cur][i]==true && dis[cur] + val[i] > dis[i] && dis[cur] + val[i] > 0 ) { dis[i]=dis[cur] + val[i]; if(!vis[i]) { q.push(i); vis[i]=true; cnt[i]++; } } } } if(dis[n]>0) return true; else { for(int k=1;k<=n;k++) for(int i=1;i<=n;i++) for(int j=1;j<=n;j++) if(map[i][k] && map[k][j]) map[i][j]=true; for(int i=1;i<=n;i++) if(cnt[i]>n && map[1][i] && map[i][n]) //忘记要从i到N了 return true; return false; } } int main() { while(~scanf("%d",&n),n!=-1) { memset(map,0,sizeof(map)); memset(val,0,sizeof(val)); for(int i=1;i<=n;i++) { int cnt,temp; scanf("%d %d",&val[i],&cnt); for(int j=1;j<=cnt;j++) { scanf("%d",&temp); map[i][temp]=true; } } if(SPFA()) puts("winnable"); else puts("hopeless"); } return 0; }