本文分享的必刷题目是从蓝桥云课、洛谷、AcWing等知名刷题平台精心挑选而来,并结合各平台提供的算法标签和难度等级进行了系统分类。题目涵盖了从基础到进阶的多种算法和数据结构,旨在为不同阶段的编程学习者提供一条清晰、平稳的学习提升路径。
欢迎大家订阅我的专栏:算法题解:C++与Python实现!
附上汇总贴:算法竞赛备考冲刺必刷题(C++) | 汇总
【题目来源】
AcWing:837. 连通块中点的数量 - AcWing题库
【题目描述】
给定一个包含 n n n 个点(编号为 1 ∼ n 1\sim n 1∼n)的无向图,初始时图中没有边。
现在要进行 m m m 个操作,操作共有三种:
C a b
,在点 a a a 和点 b b b 之间连一条边, a a a 和 b b b 可能相等;Q1 a b
,询问点 a a a 和点 b b b 是否在同一个连通块中, a a a 和 b b b 可能相等;Q2 a
,询问点 a a a 所在连通块中点的数量;
【输入】
第一行输入整数 n n n 和 m m m。
接下来
m
m
m 行,每行包含一个操作指令,指令为 C a b
,Q1 a b
或 Q2 a
中的一种。
【输出】
对于每个询问指令 Q1 a b
,如果
a
a
a 和
b
b
b 在同一个连通块中,则输出 Yes
,否则输出 No
。
对于每个询问指令 Q2 a
,输出一个整数表示点
a
a
a 所在连通块中点的数量
每个结果占一行。
【输入样例】
5 5
C 1 2
Q1 1 2
Q2 1
C 2 5
Q2 5
【输出样例】
Yes
2
3
【算法标签】
《AcWing 837 连通块中点数量》 #并查集#
【代码详解】
#include <bits/stdc++.h>
using namespace std;
const int N = 100010; // 定义最大节点数
int n, m; // n: 节点数量, m: 操作次数
int p[N]; // 并查集父节点数组
int siz[N]; // 记录每个集合的大小
// 查找x的根节点(带路径压缩优化)
int find(int x)
{
if (p[x] != x) // 如果当前节点不是根节点
p[x] = find(p[x]); // 路径压缩:将父节点设为根节点
return p[x]; // 返回根节点
}
int main()
{
scanf("%d%d", &n, &m); // 输入节点数和操作数
// 初始化并查集
for (int i = 1; i <= n; i++)
{
p[i] = i; // 每个节点自成一个集合
siz[i] = 1; // 初始集合大小为1
}
while (m--) // 处理每个操作
{
char op[5]; // 操作类型
int a, b; // 操作参数
scanf("%s", op);
if (op[0] == 'C')
{
// 合并操作:将a所在集合合并到b所在集合
scanf("%d%d", &a, &b);
if (find(a) == find(b))
continue; // 已在同一集合则跳过
siz[find(b)] += siz[find(a)]; // 合并集合大小
p[find(a)] = find(b); // 将a的根节点指向b的根节点
}
else if (op[1] == '1')
{
// 查询操作:判断a和b是否在同一集合
scanf("%d%d", &a, &b);
if (find(a) == find(b))
puts("Yes");
else
puts("No");
}
else
{
// 查询操作:返回a所在集合的大小
scanf("%d", &a);
printf("%d\n", siz[find(a)]);
}
}
return 0;
}
【运行结果】
5 5
C 1 2
Q1 1 2
Yes
Q2 1
2
C 2 5
Q2 5
3
gu