Description

给\(n\)组操作,每组操作形式为\(x\;y\;p\)。

当\(p\)为\(1\)时,如果第\(x\)变量和第\(y\)个变量可以相等,则输出\(YES\),并限制他们相等;否则输出\(NO\),并忽略此次操作。

当\(p\)为\(0\)时,如果第\(x\)变量和第\(y\)个变量可以不相等,则输出\(YES\),并限制他们不相等 ;否则输出\(NO\),并忽略此次操作。

Input

输入一个数\(n\)表示操作的次数\((n<=10^5)\)

接下来\(n\)行每行三个数\(x\;y\;p\) \((x,y<=10^8,0≤p≤1)\)

Output

对于\(n\)行操作,分别输出\(n\)行\(YES\)或者\(NO\)

Solution

没想到假的启发式合并也能A题啊。。。

正解其实跟考试时候的思路差不多

但是不是维护每个联通块的大小

因为有可能一个联通块大小比较小但是连出去的边有很多

所以我们要换一种数据结构维护每个联通块连出去了多少条边

用什么数据结构可以维护大小,快速查找两个元素是否有关系呢?

嗯... \(STL\) 的 \(set\) 是符合要求的 查询大小是 \(O(1)\) 的,查找是 \(O(nlogn)\) 的

所以我们用一个 \(set\) \(s[i]\) 表示以 \(i\) 为根的联通块连出去的边(这里连边表示规定两个联通块严格不相等)

考虑操作

如果要求两个变量相等,那么就在两个联通块的 \(set\) 里找是否存在一条边连到了对方,如果有,那么此条件无法满足。

如果要求两个变量不相等,那么假设它们不在一个联通块里,需要合并这两个联通块,就要用到启发式合并了。

注意到我们已经记录了两个联通块连出去边的个数了, 为了保证复杂度,一定是想让连边少的联通块合并到连边多的联通块里。这就是启发式合并了。

Code

#include<set>
#include<map>
#include<cstdio>
#define N 100005 int n,tot;
int ques[N][5];
int father[N<<1];
std::map<int,int> mp;
std::set<int> s[N<<1]; int find(int x){
if(father[x]==x) return x;
return father[x]=find(father[x]);
} signed main(){
scanf("%d",&n);
for(int x,y,i=1;i<=n;i++){
scanf("%d%d%d",&x,&y,&ques[i][3]);
if(!mp[x]) mp[x]=++tot;
if(!mp[y]) mp[y]=++tot;
ques[i][1]=mp[x];
ques[i][2]=mp[y];
}
for(int i=1;i<=tot;i++) father[i]=i;
for(int i=1;i<=n;i++){
int r1=find(ques[i][1]);
int r2=find(ques[i][2]);
if(ques[i][3]==1){
if(r1==r2) {puts("YES");continue;}
if(s[r1].find(r2)!=s[r1].end() or s[r2].find(r1)!=s[r2].end()){
puts("NO");
continue;
}
if(s[r1].size()>s[r2].size()) r1^=r2^=r1^=r2;
std::set<int>::iterator it;
for(it=s[r1].begin();it!=s[r1].end();it++)
s[r2].insert(*it),s[*it].insert(r2),s[*it].erase(r1);
father[r1]=r2;
puts("YES");
}
else{
if(r1==r2) {puts("NO");continue;}
if(s[r1].find(r2)!=s[r1].end() or s[r2].find(r1)!=s[r2].end()){
puts("YES");
continue;
}
s[r1].insert(r2);
s[r2].insert(r1);
puts("YES");
}
}
return 0;
}

[51nod 1515] 明辨是非的更多相关文章

  1. 51nod 1515 明辨是非 [并查集+set]

    今天cb巨巨突然拿题来问,感觉惊讶又开心,希望他早日康复!!坚持学acm!加油! 题目链接:51nod 1515 明辨是非 [并查集] 1515 明辨是非 题目来源: 原创 基准时间限制:1 秒 空间 ...

  2. 51nod 1515 明辨是非 启发式合并

    1515 明辨是非 题目连接: https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1515 Description 给n组操 ...

  3. 51Nod 1515 明辨是非 —— 并查集 + 启发式合并

    题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1515 1515 明辨是非  题目来源: 原创 基准时间限制:1 ...

  4. 51nod 1515 明辨是非 并查集 + set + 启发式合并

    给n组操作,每组操作形式为x y p. 当p为1时,如果第x变量和第y个变量可以相等,则输出YES,并限制他们相等:否则输出NO,并忽略此次操作. 当p为0时,如果第x变量和第y个变量可以不相等,则输 ...

  5. 51nod 1515 明辨是非 并查集+set维护相等与不等关系

    考试时先拿vector瞎搞不等信息,又没离散化,结果好像MLE:后来想起课上讲过用set维护,就开始瞎搞迭代器...QWQ我太菜了.. 用并查集维护相等信息,用set记录不相等的信息: 如果要求变量不 ...

  6. 51nod 1515:明辨是非 并查集合并

    1515 明辨是非 题目来源: 原创 基准时间限制:1 秒 空间限制:131072 KB 分值: 160 难度:6级算法题  收藏  关注 给n组操作,每组操作形式为x y p. 当p为1时,如果第x ...

  7. 51 nod 1515 明辨是非(并查集合并)

    1515 明辨是非题目来源: 原创基准时间限制:1 秒 空间限制:131072 KB 分值: 160 难度:6级算法题 给n组操作,每组操作形式为x y p. 当p为1时,如果第x变量和第y个变量可以 ...

  8. 【51Nod 1244】莫比乌斯函数之和

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1244 模板题... 杜教筛和基于质因子分解的筛法都写了一下模板. 杜教筛 ...

  9. 51Nod 1268 和为K的组合

    51Nod  1268  和为K的组合 1268 和为K的组合 基准时间限制:1 秒 空间限制:131072 KB 分值: 20 难度:3级算法题 给出N个正整数组成的数组A,求能否从中选出若干个,使 ...

随机推荐

  1. linux中的网络通信指令

    1.write write命令通信是一对一的通信,即两个人之间的通信,如上图. 效果图 用法:write <用户名> 2.wall wall指令可将信息发送给每位同意接收公众信息的终端机用 ...

  2. 学习Tensorflow,使用源码安装

    PC上装好Ubuntu系统,我们一步一步来讲解如何使用源码安装tensorflow?(我的Ubuntu系统是15.10) 安装cuda 根据你的系统型号选择相应的cuda版本下载 https://de ...

  3. 【shell脚本】nginx每天自动切割日志脚本

    nginx每天日志量比较大的时候,最好每天自动切割,存储,这样可以方面以后的查询和分析 #!/bin/sh ################### #filename: nginx_log_rotat ...

  4. EBS DBA指南笔记(三)

    第五章 patching   patch的作用:解决应用代码的问题:安装新的特征:更新technology stack组件.打patch不是一个简单的过程,但我们也没必要深究里面每个细节. EBS的p ...

  5. 挖掘频繁项集之FP-Growth算法

    http://blog.csdn.net/pipisorry/article/details/48918007 FP-Growth频繁项集挖掘算法(Frequent-Pattern Growth, 频 ...

  6. ValueError: setting an array element with a sequence.

    http://blog.csdn.net/pipisorry/article/details/48031035 From the code you showed us, the only thing ...

  7. 重构前VS重构后效果对比

    本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/42554641 学习重构已经一个多月了,虽然不能让代码特别的 ...

  8. Android4.2.2源码目录结构分析

    撰写不易,转载请注明出处:http://blog.csdn.net/jscese/article/details/40897277#t17 导读: 关于的Android目录分析,网上有很多资料,在此不 ...

  9. Python学习笔记 - 列表生成式listComprehensions

    #!/usr/bin/env python3 # -*- coding: utf-8 -*- list(range(1, 11)) # 生成1乘1,2乘2...10乘10 L = [] for x i ...

  10. [SqlServer]如何向数据库插入带有单引号(')的字符串

    今天在做一个复制功能的时候,发现存在单引号字符串与INSERT INTO 语句的' '产生冲突. 在网络上找到了一个这样功能 如何向数据库插入带有单引号(')的字符串 用SQL语句往数据库某字段(字符 ...