真他妈是道好题。
不知道谁能想出来拿并查集这么扯的东西。
用并查集维护一下类似前缀和的东西,如果当前是[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; }