真他妈是道好题。
不知道谁能想出来拿并查集这么扯的东西。
用并查集维护一下类似前缀和的东西,如果当前是[x,y],那么从x-1向y连一条边,v[x]表示s[x]-s[q]的值,s是前缀和,q是x所在联通块的根,如果当前的x和y是在一个联通块里,就直接查询v[y]-v[x],因为sum(x,y)=s[y]-s[q]-s[x]+s[q]=s[y]-s[x],合并的时候就v[y]=y-t,v[x]=x-s,dis=y-x,v[y]-v[x]-dis=y-t-x+s-y+x=s-t,v[s]=s-t,s、t分别是x和y的根,好扯淡。
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<iostream>
#include<algorithm>
#define maxn 1010
using namespace std;
int f[maxn],v[maxn];
int n,m,T;
bool w;
int find(int x)
{
if (f[x]==x) return x;
int q=find(f[x]);
v[x]+=v[f[x]];
f[x]=q;
return f[x];
}
void add(int x,int y,int z)
{
int p=find(x),q=find(y);
if (p!=q)
{
f[p]=q;
v[p]=v[y]-v[x]-z;
}
else if (v[y]-v[x]!=z) w=1;
}
int main()
{
scanf("%d",&T);
while (T--)
{
memset(v,0,sizeof(v));
scanf("%d%d",&n,&m);
w=0;
for (int i=0;i<=n;i++) f[i]=i;
for (int i=1;i<=m;i++)
{
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
if (!w) add(x-1,y,z);
}
if (w) printf("false\n");
else printf("true\n");
}
return 0;
}