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;
}
By ❤千柒/Kan_kiz