程序自动分析(并查集)

在实现程序自动分析的过程中,常常需要判定一些约束条件是否能被同时满足。

考虑一个约束满足问题的简化版本:假设x1,x2,x3,…
代表程序中出现的变量,给定n个形如xi=xj或xi≠xj

的变量相等/不等的约束条件,请判定是否可以分别为每一个变量赋予恰当的值,使得上述所有约束条件同时被满足。

例如,一个问题中的约束条件为:x1=x2,x2=x3,x3=x4,x1≠x4

,这些约束条件显然是不可能同时被满足的,因此这个问题应判定为不可被满足。

现在给出一些约束满足问题,请分别对它们进行判定。
输入格式

输入文件的第1行包含1个正整数t,表示需要判定的问题个数,注意这些问题之间是相互独立的。

对于每个问题,包含若干行:

第1行包含1个正整数n,表示该问题中需要被满足的约束条件个数。

接下来n行,每行包括3个整数i,j,e,描述1个相等/不等的约束条件,相邻整数之间用单个空格隔开。若e=1,则该约束条件为xi=xj
;若e=0,则该约束条件为xi≠xj


输出格式

输出文件包括t行。

输出文件的第k行输出一个字符串“YES”或者“NO”(不包含引号,字母全部大写),“YES”表示输入中的第k个问题判定为可以被满足,“NO”表示不可被满足。
数据范围

1≤n≤1000000

1≤i,j≤1000000000

输入样例:

2
2
1 2 1
1 2 0
2
1 2 1
2 1 1

输出样例:

NO
YES

**
这道题主要应用并查集来解决问题
并查集(Disjoint-Set)是一一种可以动态维护若千个不重叠的集合,并支持合并与查询的数据结构。详细地说,并查集包括如下两个基本操作:
1 Get,查询一个元素属于哪一一个集合。
2 Merge,把两个集合合并成一个大集合。
为了具体实现并查集这种数据结构,我们首先需要定义集合的表示方法。在并查集中,我们采用“代表元”法,即为每个集合选择一个固定的元素,作为整个集合的代表
1:并查集的存储
使用一个数组fa保存父节点
int fa[]
2:并查集的初始化
设有n个元素,起初所有元素各自构成个独立的集合,即有n个1个点的数。
for(int i=1;i<=n;i++)fa[i]=i;
3.并查集的Get操作
若x是树根,则x就是集合代表,否则递归访问fa[x]直至根节点。
int get(int x) {
if (x == fa[x]) return X;
return fa[x] = get(fa[x]); // 路径压缩,fa 直接赋值为代表元素
4.并查集的Merge操作
合并元素x和元素y所在的集合,等价于让x的树根作为? y的树根的子节点。void merge(int X, int y) {
fa[get(x)] = get(y);
在这道题中一开始所有的变量各自为一个集合,如果每一条都相等就合并两个变量所在的集合。
如果存在一个不等的约数条件,但是他们却存在相等关系,就不被满足
**

#include <bits/stdc++.h>
using namespace std;
const int N=1000000*2+10;
int a[N][3],n,m,i,j,cnt,t,fa[N],pan[N]; //fa表示一个查并集,pan存储的是在输入的每组数中,最后一位的值
unordered_map<int ,int> c;//因为数的范围可能会较大,所以可以将它先映射到这里
int get(int x)
{
    return fa[x]==x?x:fa[x]=get(fa[x]);//寻找他的根
}
void merge(int x,int y)
{
    fa[get(x)]=get(y);//合并
}
int main()
{
    cin>>t;
    while(t--)
    {
    cin>>n;
    n*=3;//因为每组第三个数所以要乘以三
    for(int i=1;i<=n;i++)
    {
        int x;
        cin>>x;
        if(i%3)//判断是第几个数。
        {
            if(c[x]==0)//判断这个数有没有出现过
            c[x]=(++cnt);
            a[i/3+1][i%3]=c[x];//i/3+1表示的是第几组。i%3表示的是第几个数
            fa[c[x]]=c[x];将它存入并查集中
        }
        else
        pan[i/3]=x;
    }
    int ok=0;
    for(int i=1;i<=n/3;i++)//开始遍历每组数
    {
        if(pan[i])
        merge(a[i][1],a[i][2]);//如果要是1,就合并他们
    }
    for(int i=1;i<=n/3;i++)
    if(!pan[i])
    {
        if(get(a[i][1])==get(a[i][2]))//如果是本来不应该相等的数,相等了,就返回NO
        {
            ok=true;
            cout<<"NO";break;
        }
    }
    if(!ok)
    cout<<"YES";
    cout<<endl;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
程序是一种运行在手机或其他智能设备上的应用程序,具有简洁、便捷的特点。巡检项目是指对某个设备、系统或区域进行定期检查和维护,以确保其正常运行和安全性。CSDN是一个面向IT技术人员的社区平台,提供了丰富的技术资源和交流分享的机会。 小程序巡检项目的CSDN可以提供以下几个方面的功能与服务: 发布与查询巡检任务:用户可以在小程序上发布巡检任务,包括需要巡检的设备或系统、巡检时间、巡检内容等。其他用户可以查询已发布的巡检任务,并根据自己的能力和兴趣选择参与。 巡检报告上传和管理:巡检人员可以在小程序上上传巡检报告,包括巡检结果、问题描述、照片等。系统将自动保存并分类管理这些巡检报告,方便用户查阅和分析。 问题反馈和解决方案分享:用户可以通过小程序向社区提问,寻求解决问题的方法和建议。其他技术人员可以回答问题,并分享自己的经验和解决方案。 技术资源和学习资料:小程序可以提供CSDN平台上的技术文章、教程、视频等资源,帮助用户学习和提升技术能力。用户可以根据自己的需求和兴趣进行搜索和浏览,找到适合自己的学习材料。 用户社区交流和互动:小程序可以提供用户之间的交流平台,包括讨论区、技术圈子、私聊等功能。用户可以在这里互相交流经验,分享技术问题和解决方法,建立更广泛的技术人脉。 总之,小程序巡检项目的CSDN可以提供一个集巡检任务发布与查询、巡检报告管理、问题反馈与解决方案分享、技术资源与学习资料、用户社区交流与互动等功能于一体的综合服务平台,旨在为技术人员提供更便捷、高效的巡检项目管理和学习交流的途径。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值