1202: [HNOI2005]狡猾的商人

1202: [HNOI2005]狡猾的商人

https://www.lydsy.com/JudgeOnline/problem.php?id=1202

分析

带权并查集! 首先可以把每个月抽象一个点,那么知道了a~b,a~c月的收入,相当于知道了a->b,a->c的距离。如果再知道b~c的收入,那么就可以判断了。

对应图上,相当于找到了环,判断b->c的距离是否等于a->c的距离减去a->b的距离。 a->b->c 于是并查集的功能就是找环,其次还要维护这些的距离关系。

如何维护距离?——带权并查集。 考虑每个点x附加一个值val[x]=s[x]-s[g],g为并查集的根,s为1~x的前缀和。

查询: 当前l(l默认已经减一),r已经是同一个并查集了。 查询s[r]-s[l],就是val[l]=s[l]-s[g],val[r]=s[r]-s[g],就是val[r]-val[l]=s[r]-s[l],s[g]约掉了。

合并: 新加入两个点l,r,长度为w,如果l,r已经是一个并查集,跳过。

否则,设u为l的根,v为r的根(下面默认让v为合并后的并查集的根),那么v的子树是不用变动的。

对于u的子树,先考虑u:当前val[u]=s[u]-s[u]=0,合并后val[u]=s[u]-s[v]。

正推:由val[l]=s[l]-s[u],得s[u]=val[l]+s[l],同理s[v]=val[r]+s[r] ,s[u]-s[v]=val[r]-val[l]+s[r]-s[l]=val[r]-val[l]+w;

逆推:val[l]=s[l]-s[u],val[r]=s[r]-s[v],那么用后面的减前面的得: var[r]-val[l]=s[r]-s[l]+s[u]-s[v]=w+s[u]-s[v],val[r]-val[l]-w=s[u]-s[v];

于是可以求出val[u]。 对于其他的u的子节点x,在路径压缩后,它们同样是连向v,对于x,其父节点u(并查集只有一层,即使还没有进行路径压缩,它的父节点一定是类似u的点,已更新)

val[u]=s[u]-s[v],当前val[x]=s[x]-s[u],val[x]+val[u]=s[x]-s[u]+s[u]-s[v]=s[x]-s[v]; 

代码

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
 
inline int read() {
    int x=0,f=1;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-1;
    for (;isdigit(ch);ch=getchar())x=x*10+ch-'0';return x*f;
}
 
const int N = 1010;
int fa[N],val[N];
 
int find(int x) {
    if (x == fa[x]) return x;
    int tmp = find(fa[x]);
    val[x] += val[fa[x]];
    fa[x] = tmp;
    return fa[x];
}
 
inline void work() {
    int n = read(),m = read();
    for (int i=0; i<=n; ++i) {
        fa[i] = i,val[i] = 0;
    }
    bool flag = true;
    for (int i=1; i<=m; ++i) {
        int l = read(),r = read(),w = read();
        l --;
        int u = find(l),v = find(r);
        if (u != v) {
            fa[u] = v;
            val[u] = val[r] - val[l] + w;
        }
        else if (val[l] - val[r] != w) flag = false;
    }
    puts(flag?"true":"false");
}
int main() {
    int Case = read();
    while (Case--) work();
    return 0;
}

 

posted @ 2018-10-08 16:34  MJT12044  阅读(215)  评论(0编辑  收藏  举报