银河英雄传说 解题报告

带权并查集

题目大意:
给一个1×n的序列然后每次把每一列挪到另一列后边
然后询问某两个战舰是否在同一列 如果在同一列求中间的距离
银河英雄传说 解题报告_第1张图片

首先看到合并就想起了并查集
那么距离怎么处理?
那就得加权了

对于每个集合
维护一个num[ ]记录该集合元素总数
对于每一个元素
维护一个pos[ ]记录该元素在集合里的位置(从前往后依次是0,1,2,3…)
那么每次询问中间战舰个数的时候直接输出pos[a]-pos[b]-1就行了

对于每次合并union( a , b)表示把a所在列接到b所在列后边
fa[ find(a) ]=find(b)
然后更新find(b) 的num和find(a) 的pos就可以了

对于a集合的其他元素的pos
在find时令当前元素x(除集合首个元素外)的pos等于fa[x]的pos+x本来的pos即可

但是要是多次find(a)不就把pos加很多遍了???

这是一开始担心的问题
然后发现通过路径压缩每次union的时候已经把a所在集合首个元素A的pos更新好并连到pos=0的根上了
后边查询a所在集合的其他元素的时候
由于A的pos已经被修改好并且连到根上了
所以A的pos=pos+0 也就是原a所在集合的首个元素A的pos依然正确
回到上一层 A的儿子pos+=pos[A]
这个时候
由于路径压缩的存在
然后这个儿子也被直接连到了根上 即pos=0 的位置
以后即便再次询问也不会修改掉正确的pos了 因为pos+0=pos
然后就是细节问题了 具体见注释
luogu 1196 银河英雄传说:

int find(int x)
{
    if(fa[x]==x) return fa[x];
    else {
        int f=fa[x];  //这里要先fa存下来 路径压缩之后就修改了
        fa[x]=find(fa[x]);
        pos[x]+=pos[f];
        return fa[x];
    }
}
void merge(int x,int y)
{
    int fx=find(x),fy=find(y);
    pos[fx]+=num[fy];
    fa[fx]=fy;
    num[fy]+=num[fx];
    num[fx]=0;
}

你可能感兴趣的:(洛谷)