AtCoder Beginner Contest 061 - D Score Attack

给定一张边带权的有向图。从节点\(1\)出发,每经过一条边一次,得分加上这条边的边权。(可以多次经过,多次累加

必须在点\(n\)结束游戏

判断是否能使得分无限大,如果否,求最大得分。

sol

题目所给的边权可以为负,\(dfs\)判环+\(DAG\)\(DP\)最长路是补锌的(?

然后很容易想到边权取相反数,跑一遍\(SPFA\)判负环+求最短路。哇,sb题,秒了

然后喜提WA QAQ

被坑了————————————————

\(SPFA\)不可行的原因是,题目要求路径从节点\(1\)到节点\(n\)。判出负环就输出\(inf\)会少考虑一种情况,那就是负环并不在\(1\)\(n\)的路径上。

所以我们用\(BellmanFord\)

先进行\(n-1\)轮松弛,记录此时的\(dis[n]\)

再进行\(n-1\)轮松弛()新的\(dis[n]\)发生了变动,当且仅当在\(1\)\(n\)的路径上存在负环。

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cmath>
using namespace std;
int n,m;
#define MAXN (2007)
#define MAXM (4014)
struct qwq
{
	int nex,to,w; 
}e[MAXM];
int h[MAXN],tot=0;
inline void add(int x,int y,int z)
{
	e[++tot].to=y;
	e[tot].w=z;
	e[tot].nex=h[x];
	h[x]=tot;
}
long long dis[MAXN];
int coun[MAXN];
bool vis[MAXN];
#include <queue>
#define inf (long long)((long long)(1e16)+233)
queue<int> q;
inline void INIT() { for (int i=1;i<=n;i++) dis[i]=inf; }
inline long long spfa()//关于SPFA,_____________
{
	INIT();
	q.push(1);
	vis[1]=true;
	dis[1]=0;
	int x;
	while (!q.empty())
	{
		x=q.front();
		vis[x]=false;
		q.pop();
		for (int i=h[x],y;i;i=e[i].nex)
		{
			y=e[i].to;
			
//			printf("x:%d dis:%lld  y:%d\n",x,dis[x],y);
			if (dis[y]>dis[x]+e[i].w)
			{
				dis[y]=dis[x]+e[i].w;
				if (!vis[y])
				{
					coun[y]=coun[x]+1;
					if (coun[y]>=n) return (inf<<1);
					vis[y]=true;
					q.push(y);
				}
			}
		}
	}
	return dis[n];
}
inline long long BF()
{
	INIT();
	dis[1]=0;
	for (int k=1;k<n;k++)
		for (int x=1;x<=n;x++)
			for (int i=h[x],y;i;i=e[i].nex)
			{
				y=e[i].to;
				
				if (dis[y]>dis[x]+e[i].w) dis[y]=dis[x]+e[i].w;
			}
	long long ans=dis[n];
	for (int k=1;k<n;k++)
		for (int x=1;x<=n;x++)
			for (int i=h[x],y;i;i=e[i].nex)
			{
				y=e[i].to;
				if (dis[y]>dis[x]+e[i].w) dis[y]=dis[x]+e[i].w;
			}
	if (ans!=dis[n]) return inf<<1;
	else return ans;
}
signed main()
{
	scanf("%d%d",&n,&m);
	for (int i=1,x,y,z;i<=m;i++)
	{
		scanf("%d%d%d",&x,&y,&z);
		add(x,y,-z);
	}
	long long ans=BF();
	if (ans==inf<<1) puts("inf");
	else printf("%lld\n",-ans);
	return 0;
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章