【bzoj1202】 HNOI2005狡猾的商人 并查集

真他妈是道好题。

不知道谁能想出来拿并查集这么扯的东西。

用并查集维护一下类似前缀和的东西,如果当前是[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;
}


你可能感兴趣的:(【bzoj1202】 HNOI2005狡猾的商人 并查集)