《洛谷深入浅出进阶篇》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;
	}
}

  • 5
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

louisdlee.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值