算法:查并集

#预处理 #基础算法

维护"集合"的三件套:查询/合并/判同

普通并查集

数据结构

  • fai:结点i的父指针,根结点指向自己
  • 初始化:每个元素自成集合,fai=i;

三大基础操作

cpp 复制代码
void init(int n) { for(int i=1;i<=n;i++) fa[i]=i; }

int find(int x) {                    // 路径压缩
    if(fa[x]==x) return x;
    return fa[x]=find(fa[x]);
}

void un(int x,int y) {               // 按秩/大小合并可写一行
    int fx=find(x), fy=find(y);
    if(fx!=fy) fa[fx]=fy;            // 把 fx 挂到 fy 下
}

bool same(int x,int y) { return find(x)==find(y); }

统计集合个数

cpp 复制代码
int cnt=0;
for(int i=1;i<=n;i++) if(fa[i]==i) cnt++;

扩展域查并集(关系>2种)

  • 核心思想:把"每个元素"拆成多个域,每个域代表一种状态/关系,用偏移量存储。

经典场景:朋友&敌人(2倍域)

  • 域1~n: 朋友身份
  • 域n+1~2n:敌人身份

合并规则

  • 朋友 x,y → un(x,y) 且 un(x+n,y+n)
  • 敌人 x,y → un(x,y+n) 且 un(y,x+n)

代码模板

cpp 复制代码
const int N = 1e6 + 10;
int fa[N * 2];             // 0~n-1 为原域,n~2n-1 为"敌人域"

void init(int n) {
    for (int i = 1; i <= 2 * n; i++) fa[i] = i;
}

int find(int x) {          // 与普通版完全相同
    if (fa[x] == x) return x;
    return fa[x] = find(fa[x]);
}

void un(int x, int y) {    // 合并根,普通写法
    int fx = find(x), fy = find(y);
    if (fx != fy) fa[fx] = fy;
}

void mergeFriend(int x, int y, int n) { // 朋友:双向绑定
    un(x, y);
    un(x + n, y + n);
}

void mergeEnemy(int x, int y, int n) {  // 敌人:交叉绑定
    un(x, y + n);
    un(y, x + n);
}

bool isFriend(int x, int y) { return find(x) == find(y); }
bool isEnemy (int x, int y)  { return find(x) == find(y + n); }
bool conflict(int x, int y, int n) { // 判断是否"既友又敌"
    return isFriend(x, y) && isEnemy(x, y, n);
}

带权并查集

核心思想:给每条父边记录"权值 di",表示 i 到父节点的某种累积量(距离、差值、高度差等)。路径压缩后 di 直接是 i → 根 的累积。

数据结构

cpp 复制代码
int fa[N], d[N];   // d[i]:i 相对于父节点的权值
void init(int n) {
    for(int i=1;i<=n;i++) { fa[i]=i; d[i]=0; }
}

查询(路径压缩+累计权值)

cpp 复制代码
int find(int x) {
    if(fa[x]==x) return x;
    int root=find(fa[x]);   // 先让祖先挂到根
    d[x] += d[fa[x]];       // 累加"祖父→根"部分
    return fa[x]=root;      // 挂根,返回根
}

合并

(按题目给出的权值公式)

  • 以 "x 到 y 的边权为 w" 为例(x 在 fx 树,y 在 fy 树)
  • 目标:让 fx 挂到 fy,且满足dx + dfx + w == dy→ dfx = dy - dx - w
cpp 复制代码
void un(int x,int y,int w) {   // w: x→y 的给定边权
    int fx=find(x), fy=find(y);
    if(fx==fy) return;         // 已同集合,视情况判断矛盾
    fa[fx]=fy;
    d[fx] = d[y] - d[x] - w;   // 保证公式成立
}

查询两点间量值

cpp 复制代码
int query(int x,int y) {
    int fx=find(x), fy=find(y);
    if(fx!=fy) return INF;     // 不同集合,量值未知
    return d[y] - d[x];        // 根据合并公式推导
}

选用策略

场景 用法
纯连通、集合计数 普通 UFS + 路径压缩 + 按秩合并
朋友-敌人 / 食物链 扩展域(2 倍或 3 倍)
差分、距离、相对高度 带权 UFS(维护 d\[\])
相关推荐
yaoxin5211232 小时前
434. Java 日期时间 API - Period 基于日期的时间段
java·开发语言·python
凡人叶枫2 小时前
Effective C++ 条款30:透彻了解 inlining 的里里外外
linux·开发语言·c++·嵌入式开发·effective c++
noipp2 小时前
推荐题目:洛谷 P10907 [蓝桥杯 2024 国 B] 蚂蚁开会
c语言·c++·算法·编程·洛谷
学逆向的3 小时前
C++纯虚函数
开发语言·c++·网络安全
程序员二叉3 小时前
【JUC】ThreadLocal底层原理|内存泄漏|弱引用|跨线程传递方案
java·开发语言·面试·职场和发展·juc
程序员二叉3 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
青山木3 小时前
Hot 100 --- 轮转数组
java·数据结构·算法
徐小夕4 小时前
Loop Engineering 深度解析与实战指南(全网最全)
前端·算法·github
凡人叶枫4 小时前
Effective C++ 条款22:将成员变量声明为 private
linux·开发语言·c++
Qt程序员4 小时前
掌握 Linux 内核调度:从原理到实现(进程篇)
java·开发语言