并查集进阶版


过关代码如下

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;
}
相关推荐
码破苍穹ovo4 小时前
堆----1.数组中的第K个最大元素
java·数据结构·算法·排序算法
愤怒的小鸟~~~4 小时前
c语言创建的一个队列结构(含有这个头指针和这个尾指针的结构具有一定的参考价值)
c语言·开发语言·算法
Joker-01115 小时前
深入 Go 底层原理(十二):map 的实现与哈希冲突
算法·go·哈希算法·map
金融小师妹6 小时前
AI量化模型解析黄金3300关口博弈:市场聚焦“非农数据”的GRU-RNN混合架构推演
大数据·人工智能·算法
金融小师妹6 小时前
基于LSTM-GRU混合网络的动态解析:美联储维稳政策与黄金单日跌1.5%的非线性关联
大数据·人工智能·算法
白日梦想家-K7 小时前
题单【模拟与高精度】
开发语言·c++·算法
重生之我是Java开发战士7 小时前
【C语言】内存函数与数据在内存中的存储
c语言·开发语言·算法
roman_日积跬步-终至千里8 小时前
【机器学习】“回归“算法模型的三个评估指标:MAE(衡量预测准确性)、MSE(放大大误差)、R²(说明模型解释能力)
算法·机器学习·回归
小指纹10 小时前
图论-最短路Dijkstra算法
数据结构·c++·算法·深度优先·图论
皮蛋瘦肉粥_12110 小时前
代码随想录day52图论3
图论