当前位置: 首页 > article >正文

《洛谷深入浅出进阶篇》P1995 程序自动分析——并查集,离散化

上链接:P1955 [NOI2015] 程序自动分析 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)icon-default.png?t=N7T8https://www.luogu.com.cn/problem/P1955

上题干:

首先给你一个整数t,代表t次操作。

每一次操作包含以下内容:

1.给你一个整数n,让你执行n次条件

接下来n行,每行给你3个整数i,j,k,

例如 1 2 1

或  1 2 0

(前面两个数代表下标,第三个整数k代表条件,如果它是1 ,则代表 x1 = x2,如果它是0,代表x1!=x2) 

然后我们每一次操作需要输出yes 或 no 来表示这n个条件是否全部成立。

(假如给出这样的数据:

    n=4

    1 2 1

    2 3 1

    1 4 1

    3 4 0

    第一个条件到第四个条件分别是 x1=x2,x2=x3,x1=x4 ,x3!=x4

由于前面三个条件我们可以得知  x1=x2=x3=x4   所以与x3!=x4矛盾,输出no)

数据 n<=1e6, i,j<=1e9

很显然,这道题的条件和数据范围告诉我们:需要用到并查集+(离散化 or hash表)

第一,并差集的特点就是能将不同的集合连接到一起,然后便于我们查询某两个元素是否为同一集合。

第二,这道题的数据范围太大了,i能达到ie9,所以我们直接用并查集的话,毫无疑问,数组装不下。所以我们可以用离散化来大大缩小数据范围,除此之外呢,我们还可以用hash表来处理这样的大数据,使得每一个大数据都有一个特别的标记与之对应。

这两种方法都满足我们的需求,这里主要用离散化来实现代码。

还记得离散化的具体步骤吗?

记录散点,排序,去重,二分查找

并查集的具体步骤呢?

初始化集合,建立查询函数,合并函数。

所以我们的思路是这样的:

1,先将每一次的散点都存入一个数组b中

2,对这个数组b进行排序

3,对这个数组进行去重(可以选择重新建立一个数组c来存放去重后的数据,也可以直接用unique函数。)

4,二分查找每一对散点的相对位置。

5,初始化并查集

6,如果第三个数字k=1,我们就利用并查集来合并两个集合。

7,如果第三个数字k=0,我们就查询两个数是否为同一集合,如果是同一集合,那么我们有

上代码:


const int MAXN = 1e6 + 10;
struct st {
	int x, y, z;
};
st a[MAXN];//三组输入数据存放之处
int b[2 * MAXN];// 存入散点
int c[2 * MAXN];//排序数组
int fa[2 * MAXN];//并查集
int btop, ctop;

int find1(int x)
{
	if (x == fa[x])return fa[x];
	return fa[x] = find1(fa[x]);
}
void join1(int c1, int c2)
{
	int f1 = find1(c1), f2 = find1(c2);
	if (f1 != f2)fa[f1] = f2;
}
int main()
{
	int t;
	cin >> t;
	while (t--)
	{
		btop = 0;
		ctop = 0;
		int n;
		cin >> n;
		for (int i = 1; i <= n; i++)
		{
			cin >> a[i].x >> a[i].y >> a[i].z;
			b[++btop] = a[i].x;
			b[++btop] = a[i].y;
		}
		sort(b + 1, b + 1 + btop);
		for (int i = 1; i <= btop; i++)if (i == 1 or b[i] != b[i - 1])c[++ctop] = b[i];
		for (int i = 1; i <= n; i++)
		{
			a[i].x = lower_bound(c + 1, c + 1 + ctop, a[i].x) - c;
			a[i].y = lower_bound(c + 1, c + 1 + ctop, a[i].y) - c;
		}
		for (int i = 1; i <= ctop; i++)
		{
			fa[i] = i;
		}
		for (int i = 1; i <= n; i++)
		{
			if(a[i].z)
			join1(a[i].x, a[i].y);
		}
		bool fk = 1;
		for (int i = 1; i <= n; i++)
		{
			if (a[i].z==0)
			{
				if (find1(a[i].x) == find1(a[i].y))
					fk = 0;
			}
		}
		if (fk == 0)cout << "NO" << endl;
		else cout << "YES" << endl;
	}
}


http://www.kler.cn/a/134159.html

相关文章:

  • 【LeetCode】【算法】581. 最短无序连续子数组
  • 华为大变革?仓颉编程语言会代替ArkTS吗?
  • 关于GCC内联汇编(也可以叫内嵌汇编)的简单学习
  • Flink_DataStreamAPI_输出算子Sink
  • Qt 获取当前系统中连接的所有USB设备的信息 libudev版
  • vue el-date-picker 日期选择器禁用失效问题
  • 全新酷盒9.0源码:多功能工具箱软件的最新iapp解决方案
  • 【计算思维】蓝桥杯STEMA 科技素养考试真题及解析 4
  • 【NGINX--1】基础知识
  • 重磅 | 进一步夯实生态建设,朗思科技与阿里龙蜥完成兼容性认证
  • 【前端知识】Node——使用fs模块对文件、文件夹的操作
  • 【kafka】使用docker启动kafka
  • 2024年山东省职业院校技能大赛中职组 “网络安全”赛项竞赛试题-B卷
  • 【C++历练之路】list的重要接口||底层逻辑的三个封装以及模拟实现
  • Kubernetes学习-概念2
  • 你知道什么是SaaS吗?
  • 在Linux上安装Oracle 数据库 11g (含静默方式安装)
  • 浅析AcrelEMS-CIA机场智慧能源管平台解决方案-安科瑞 蒋静
  • Kafka 集群实现数据同步
  • Spring 配置
  • 【案例】可视化大屏
  • 度加创作工具 演示
  • 深入理解注意力机制(下)——缩放点积注意力及示例
  • 前端uniapp列表下拉到底部加载下一页列表【下拉加载页面/带源码/实战】
  • 【洛谷 P1182】数列分段 Section II 题解(二分答案+循环)
  • WSL 2 更改默认安装的 Linux 发行版