usaco的月赛题。
记录两个点之间x方向和y方向的相对距离,用并查集维护。
若与poj 1182食物链进行比较,便会发现路径压缩部分,集合合并部分的相似点。
所以并查集不难,是有一定套路可循的。大家一定要好好总结。
【代码】
#include <iostream>
#include <cstring>
#include <string>
#include <cstdio>
#include <algorithm>
using namespace std;
const int N=40005,K=10005;
struct node
{
int x,y,idx,n;
}a[K];
int x[N],y[N],dx[N],dy[N],rx[N],ry[N],ans[K],fa[N];
int n,m,k;
int find(int x)
{
if (fa[x]==x) return x;
int t=fa[x];
fa[x]=find(fa[x]);
rx[x]+=rx[t];
ry[x]+=ry[t];
return fa[x];
}
int cmp(node a,node b)
{
return a.idx<b.idx;
}
int main()
{
int i,j,d,fx,fy;
char c;
freopen("in","r",stdin);
scanf("%d%d",&n,&m);
for (i=1;i<=n;i++)
{
fa[i]=i;
rx[i]=ry[i]=0;
}
for (i=1;i<=m;i++)
{
scanf("%d%d%d %c",&x[i],&y[i],&d,&c);
switch(c)
{
case 'W':dx[i]=-d;dy[i]=0;break;
case 'S':dx[i]=0;dy[i]=-d;break;
case 'E':dx[i]=d;dy[i]=0;break;
case 'N':dx[i]=0;dy[i]=d;break;
}
}
scanf("%d",&k);
for (i=1;i<=k;i++)
{
scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].idx);
a[i].n=i;
}
sort(a+1,a+k+1,cmp);
j=1;
for (i=1;i<=k;i++)
{
for (;j<=a[i].idx;j++)
{
fx=find(x[j]);fy=find(y[j]);
fa[fy]=fx;
rx[fy]=rx[x[j]]-rx[y[j]]-dx[j];
ry[fy]=ry[x[j]]-ry[y[j]]-dy[j];
}
if (find(a[i].x)!=find(a[i].y))
ans[a[i].n]=-1;
else
ans[a[i].n]=abs(rx[a[i].x]-rx[a[i].y])+abs(ry[a[i].x]-ry[a[i].y]);
}
for (i=1;i<=k;i++)
printf("%d\n",ans[i]);
}