并查集及其应用

简介: 并查集及其应用

1.格子游戏

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

#include<bits/stdc++.h>
using namespace std;
const int N=40010;
int p[N];
int n,m;
int find(int x)
{
    if(p[x]!=x) p[x]=find(p[x]);
    return p[x];
}
int get(int x,int y)//二维坐标映射到一维
{
    return x*n+y;
}
int main()
{
   cin>>n>>m;
   for(int i=0;i<N;i++) p[i]=i;
   int res=0;
   for(int i=1;i<=m;i++)
   {
       int x,y,a,b;
       char type;
       cin>>x>>y>>type;
       x--,y--;//因为下标是从1开始的,映射到从0开始
       a=get(x,y);//把二位坐标映射到一维里
       if(type=='D') b=get(x+1,y);
       else b=get(x,y+1);
       int pa=find(a),pb=find(b);
       if(pa==pb)//假如有环的
       {
           res=i;
           break;
       }
       p[pa]=pb;
   }
   if(res) cout<<res<<endl;
   else puts("draw");
    return 0;
}

2.搭配购买

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

把搭配购买的看成一个整体,然后做一遍01背包就可

 

#include<bits/stdc++.h>
using namespace std;
const int N=1e4+10;
int p[N],v[N],w[N];
int n,m,vol;
int f[N];
int find(int x)
{
    if(p[x]!=x) p[x]=find(p[x]);
    return p[x];
}
int main()
{
   cin>>n>>m>>vol;
   for(int i=1;i<=n;i++) p[i]=i;
   for(int i=1;i<=n;i++) cin>>v[i]>>w[i];
   while(m--)
   {
       int a,b;
       cin>>a>>b;
       a=find(a),b=find(b);
       if(a!=b)
       {
           v[b]+=v[a];//把a的价钱绑定在b中
           w[b]+=w[a];//把a的价值绑定在b中
           p[a]=b;//把ad节点合并在b中
       }
   }
   for(int i=1;i<=n;i++)//做一遍01背包
        if(p[i]==i)//假如是根节点
           for(int j=vol;j>=v[i];j--)
             f[j]=max(f[j],f[j-v[i]]+w[i]);
  cout<<f[vol]<<endl;
    return 0;
}

3.程序自动分析

237. 程序自动分析 - AcWing题库

把点离散化了,然后相等就和合并同个集合,在判断不相等有没有矛盾即可

#include<bits/stdc++.h>
using namespace std;
const int N=2e5+10;
int p[N],n;
struct Q//存询问
{
    int a,b,e;
}q[N];
unordered_map<int,int> ha;
int find(int x)
{
    if(p[x]!=x) p[x]=find(p[x]);
    return p[x];
}
int get(int x)//把坐标进行映射
{
    if(!ha.count(x)) ha[x]=++n;
    return ha[x];
}
bool solve()
{
    n=0;
    int k;
    scanf("%d",&k);
    ha.clear();//清空上一层的状态
    for(int i=1;i<=k;i++) 
    {
        int a,b,e;
        scanf("%d%d%d",&a,&b,&e);
        q[i]={get(a),get(b),e};
    }
    for(int i=1;i<=n;i++) p[i]=i;//初始化
    for(int i=1;i<=k;i++)
        if(q[i].e)//假如是相等就合并
        {
            int a=find(q[i].a),b=find(q[i].b);
            p[a]=b;
        }
    for(int i=1;i<=k;i++)
        if(!q[i].e)//假如不相等看看在不在一个集合里,假如在则矛盾
        {
            int a=find(q[i].a),b=find(q[i].b);
             if(a==b) return false;
        }
    return true;//反之没矛盾
}
int main()
{
  int T;
  scanf("%d",&T);
  while(T--)
     if(solve()) puts("YES");
     else puts("NO");
    return 0;
}

4.奇偶游戏

239. 奇偶游戏 - AcWing题库

1.带边权的做法

异或就是二进制不进位加法

dx表示跟根节点的关系

#include<bits/stdc++.h>
using namespace std;
const int N=1e4+10;
int p[N],d[N];//s表示集合的长度,d表示每个点到根节点的距离
int n,m;
unordered_map<int,int> ha;
int get(int x)
{
    if(!ha.count(x)) ha[x]=++n;
    return ha[x];
}
int find(int x)
{
    if(p[x]!=x) 
    {
        int root=find(p[x]);
        d[x]^=d[p[x]];//更新路径长其他的值
        p[x]=root;
    }
    return p[x];
}
int main()
{
    cin>>n>>m;
    n=0;
    int res=m;
    for(int i=0;i<N;i++) p[i]=i;//初始化
    for(int i=1;i<=m;i++)
    {
        int a,b;
        string type;
        cin>>a>>b>>type;
        a=get(a-1),b=get(b);//获取离散化后的值
        int t=0;
        if(type=="odd") t=1;//假如是奇数
        int pa=find(a),pb=find(b);
        if(pa==pb)//假如在一个集合
        {
          if((d[a]^d[b])!=t)//假如跟t不同类
          {
             res=i-1;
             break;
          }
        }
        else//反之不在一个集合就合并
        {
            p[pa]=pb;
            d[pa]=d[a]^d[b]^t;//合并到同一类中
        }
    }
    cout<<res<<endl;
    return 0;
}

2. 用扩展域的做法

假如条件少的话可以用,因为这里只有奇数和偶数,两个条件,可以用

 

#include<bits/stdc++.h>
using namespace std;
const int N=2e4+10,M=N/2;//M是分界线在0-M是偶数,大于M是奇数
int p[N],d[N];//s表示集合的长度,d表示每个点到根节点的距离
int n,m;
unordered_map<int,int> ha;
int get(int x)
{
    if(!ha.count(x)) ha[x]=++n;
    return ha[x];
}
int find(int x)
{
    if(p[x]!=x) p[x]=find(p[x]);
    return p[x];
}
int main()
{
    cin>>n>>m;
    n=0;
    int res=m;
    for(int i=0;i<N;i++) p[i]=i;//初始化
    for(int i=1;i<=m;i++)
    {
        int a,b;
        string type;
        cin>>a>>b>>type;
        a=get(a-1),b=get(b);//获取离散化后的值
        int a1=find(a),a2=find(a+M),b1=find(b),b2=find(b+M);//获取a跟b的偶数和奇数情况
        if(type=="odd")//假如是奇数类型
        {
            if(a1==b1||a2==b2)//但是a跟b同类了
            {
                res=i-1;
                break;
            }
            p[a1]=b2;//反之把a的偶数合并到b的奇数
            p[a2]=b1;//反之把a的奇数合并到b的偶数
        }
        else//假如是偶数
        {
            if(a1==b2||a2==b1)//假如不同类
            {
                res=i-1;
                break;
            }
            p[a1]=b1;//把偶数合并到偶数里
            p[a2]=b2;//把奇数合并到奇数里
        }
    }
    cout<<res<<endl;
    return 0;
}

5.银河英雄传说

238. 银河英雄传说 - AcWing题库

#include<bits/stdc++.h>
using namespace std;
const int N=3e4+10;
int p[N],d[N],s[N];//s表示集合的长度,d表示每个点到根节点的距离
int find(int x)
{
    if(p[x]!=x)
    {
        int root=find(p[x]);//先弄出根
        d[x]+=d[p[x]];//把路径长的长度都加上,给这个点
        p[x]=root;//最后让p[x]等于根
    }
    return p[x];
}
int main()
{
    int m;
    scanf("%d",&m);
    for(int i=1;i<N;i++) p[i]=i,s[i]=1;
    while(m--)
    {
        char op[2];
        int a,b;
        scanf("%s%d%d",op,&a,&b);
        int pa=find(a),pb=find(b);
        if(op[0]=='M')
        {
            if(pa==pb) continue;//假如已经合并过了
            d[pa]=s[pb];//让根+pb的长度
            s[pb]+=s[pa];//更新新的b的长度
            p[pa]=pb;//把a合并到b里
        }
        else
        {
            if(pa!=pb) puts("-1");//假如不在一列中
            else printf("%d\n",max(0,abs(d[a]-d[b])-1));//假如在一列中输出距离
        }
    }
    return 0;
}
相关文章
|
数据可视化 测试技术 PyTorch
智谱ChatGLM3魔搭最佳实践教程来了!
ChatGLM3-6B 是 ChatGLM 系列最新一代的开源模型,在保留了前两代模型对话流畅、部署门槛低等众多优秀特性的基础上
|
机器学习/深度学习 Shell Linux
shell基础学习
shell基础学习
133 0
|
JavaScript
vue尚品汇商城项目-day07【vue插件-49.(了解)自定义插件】
vue尚品汇商城项目-day07【vue插件-49.(了解)自定义插件】
137 0
|
存储 C语言
【C语言】浮点型数据在内存中的存储
⭐浮点数在内存中的存储 float、double、long double 浮点数的范围定义在float.h头文件中
|
5天前
|
人工智能 自然语言处理 JavaScript
2026年Windows+Ollama本地部署OpenClaw保姆级教程:本地AI Agent+阿里云上快速搭建
2026年OpenClaw凭借本地部署、私有化运行的特性,成为打造个人智能体的核心工具,而Ollama作为轻量级本地大模型管理工具,能让OpenClaw摆脱对云端大模型的依赖,实现**本地推理、数据不泄露、全流程私有化**的智能体验。本文基于Windows 11系统,从硬件环境准备、Ollama安装与模型定制、OpenClaw部署配置、技能扩展到常见问题排查,打造保姆级本地部署教程,同时补充阿里云OpenClaw(Clawdbot)快速部署步骤,兼顾本地私有化需求与云端7×24小时运行需求,文中所有代码命令均可直接复制执行,确保零基础用户也能快速搭建属于自己的本地智能体。
6884 47
|
2天前
|
人工智能 安全 API
CoPaw:3分钟部署你的 AI助理
源自阿里巴巴开源生态的个人 AI 助理——CoPaw。作为阿里倾力打造的开源力作,CoPaw 完美打通钉钉、飞书、Discord 等多平台对话通道,支持定时任务自动化。内置 PDF/Office 深度处理、新闻摘要等强大技能,更开放自定义扩展接口。坚持数据全程私有化部署,绝不上传云端,让每一位用户都能在大厂技术加持下,拥有安全、专属的智能助手。
|
6天前
|
人工智能 JSON JavaScript
手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人
手把手教你用 OpenClaw(v2026.2.22-2)+ 飞书,10分钟零代码搭建专属AI机器人!内置飞书插件,无需额外安装;支持Claude等主流模型,命令行一键配置。告别复杂开发,像聊同事一样自然对话。
3248 9
手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人
|
4天前
|
人工智能 自然语言处理 机器人
保姆级教程:Mac本地搭建OpenClaw及阿里云上1分钟部署OpenClaw+飞书集成实战指南
OpenClaw(曾用名Clawdbot、Moltbot)作为2026年最热门的开源个人AI助手平台,以“自然语言驱动自动化”为核心,支持对接飞书、Telegram等主流通讯工具,可替代人工完成文件操作、日历管理、邮件处理等重复性工作。其模块化架构适配多系统环境,既可以在Mac上本地化部署打造私人助手,也能通过阿里云实现7×24小时稳定运行,完美兼顾隐私性与便捷性。
2675 4
|
12天前
|
存储 人工智能 负载均衡
阿里云OpenClaw多Agent实战宝典:从极速部署到AI团队搭建,一个人=一支高效军团
在AI自动化时代,单一Agent的“全能模式”早已无法满足复杂任务需求——记忆臃肿导致响应迟缓、上下文污染引发逻辑冲突、无关信息加载造成Token浪费,这些痛点让OpenClaw的潜力大打折扣。而多Agent架构的出现,彻底改变了这一现状:通过“单Gateway+多分身”模式,让一个Bot在不同场景下切换独立“大脑”,如同组建一支分工明确的AI团队,实现创意、写作、编码、数据分析等任务的高效协同。
5319 31