并查集进阶版


过关代码如下

cpp 复制代码
#define _CRT_SECURE_NO_WARNINGS
#include<bits/stdc++.h>
#include<unordered_set>
using namespace std;

int n, m;
vector<int> edg[400005];
int a[400005], be[400005]; // a的作用就是存放要摧毁
int k;
int fa[400005];
int daan[400005];

void add(int x, int y) {
	edg[x].push_back(y);
	edg[y].push_back(x);
}

int find(int x) {
	if (x == fa[x]) return x;
	return fa[x] = find(fa[x]);
}

void uni(int x, int y) {
	int xx = find(x), yy = find(y);
	if (xx == yy) return;
	fa[xx] = yy;
}

int main() {
	cin >> n >> m;
	int l, r;
	for (int i = 1; i <= m; i++) {
		cin >> l >> r;
		add(l, r); // 建立边
	}
	cin >> k;
	for (int i = 1; i <= k; i++) {
		cin >> a[i];
		be[a[i]] = 1; // 标记为1,表示被摧毁
	}
	int ans = n-k; // 一开始的时候每个点都是一个块
	// 初始化并查集
	for (int i = 0; i <= n; i++) fa[i] = i;
	// 开始区分联通分量
	for (int i = 0; i < n; i++) {
		if (be[i]) continue;
		for (int u : edg[i]) {
			if (be[u]) continue;
			if (find(i) == find(u)) continue;
			uni(i, u);// 连接
			ans--;
		}
	}
	daan[k+1] = ans;
	for (int i = k; i >= 1; i--) {
		//cout << "yes" << endl;
		int xiufu = a[i];
		ans++;
		be[xiufu] = 0;  // 恢复为1
		for (int u : edg[xiufu]) {
			if (be[u]) continue;
			if (find(u) == find(xiufu)) continue;
			ans--;
			uni(u, xiufu);
		}
		daan[i] = ans;
	}
	for (int i = 1; i <= k+1; i++) {
		cout << daan[i] << endl;
	}
	//cout << " now";
	return 0;
}
相关推荐
CoderYanger8 小时前
动态规划算法-简单多状态dp问题:14.粉刷房子
开发语言·算法·leetcode·动态规划·1024程序员节
张张努力变强8 小时前
二叉树——精选题目,体验递归的暴力美学!
c语言·数据结构·算法
FMRbpm8 小时前
栈练习--------(LeetCode 739-每日温度)
数据结构·c++·算法·leetcode·新手入门
子一!!8 小时前
数据结构==二叉平衡树,AVL树 ===
数据结构·算法
Mr_Oak8 小时前
【multi-model】DINOv2(包含iBOT)& 问答
图像处理·人工智能·深度学习·算法·多模态·对比学习·视觉大模型
七夜zippoe8 小时前
轻量模型微调:LoRA、QLoRA实战对比与工程实践指南
人工智能·深度学习·算法·lora·qlora·量化训练
youngee118 小时前
hot100-44从前序与中序遍历构造二叉树
数据结构·算法
im_AMBER8 小时前
Leetcode 68 搜索插入位置 | 寻找比目标字母大的最小字母
数据结构·笔记·学习·算法·leetcode
严文文-Chris8 小时前
【非监督学习常见算法】
学习·算法·机器学习
CoderYanger9 小时前
动态规划算法-斐波那契数列模型:1.第N个泰波那契数
开发语言·算法·leetcode·动态规划·1024程序员节