当前位置:网站首页>并查集介绍和基于并查集解决问题——LeetCode 952 按公因数计算最大组件大小
并查集介绍和基于并查集解决问题——LeetCode 952 按公因数计算最大组件大小
2022-08-04 07:28:00 【dor.yang】
题目内容
给定一个由不同正整数的组成的非空数组 nums ,考虑下面的图:
有 nums.length 个节点,按从 nums[0] 到 nums[nums.length - 1] 标记;
只有当 nums[i] 和 nums[j] 共用一个大于 1 的公因数时,nums[i] 和 nums[j]之间才有一条边。
返回 图中最大连通组件的大小 。
示例 1:
输入:nums = [4,6,15,35]
输出:4
示例 2:
输入:nums = [20,50,9,63]
输出:2
示例 3:
输入:nums = [2,3,6,7,4,12,21,39]
输出:8
提示:
1 <= nums.length <= 2 * 104
1 <= nums[i] <= 105
nums 中所有值都 不同
来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/largest-component-size-by-common-factor
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解题思路
讲真,这道题一开始很难没思路,至少这个暴力是看的见的,不过很明显的一个问题就是数据规模太大了,这个规模我们要每一个求因子,然后两两一组去比是必不能行的,后来我们考虑了哈希表,感觉也不太行,于是就依据题解选择了并查集。
这是我之前学习并查集时候的笔记(学完了真的,要code一下,不然害人害己 T^T)
并查集,常见重要的一个结构
功能1:返回两个元素是否在一个集合中。
功能2:集合的合并
这个功能实现好像还好是吧?但是实际上我们实现出来的时间复杂度八成会超标(链表功能1不好,哈希表功能2不好,时间比较久),这就是并查集的大优势,他对于上面两个功能都是O(1)级别的。
并查集功能1思路:一直回溯,以第一个元素作为标志。如果两个元素去查看,发现标志元素一致,就说明在一个集合里,反之不在。
功能2思路:先功能1,之后合并就是简单的挂靠,可以想象成链表拼接那种感觉,不过更像是树或者说图树枝的延伸。一般是少顶的挂在多的顶部底下。
并查集为什么那么好,因为他的实现还有优化。
功能1:本意是要我们一直往上,找到头去对比,不过我们第一个元素走到头之后,第二个元素的时候就不用重复走第一个元素的路了,我们把第一个元素走过的路都扁平化,也就是途中节点直接走到终点。
然后我们大概了解了,并查集的功能了,我们来看下这道题是如何通过并查集来实现的。这里我们也是通过画图来看一下:
就是说,我们是通过其他节点以我们nums的数据作为代表节点进行合并操作的方式实现的两个节点之间的联系,同时完成更新代表节点,不过这个时候我心里有一个想法就是修改并查集的合并机制,我让所有新来的节点作为头,叫大子树拼接过来,然后统计rank-1。然后,果不其然,这个是有问题的,并且问题多到不知能从何开始说起了,主要原因我觉得是在于节点2,2这个节点的因子判断会导致一些问题,具体的。。我写代码的时候遇到了一些杂七杂八的,三言两语也说不太清,还是走并查集正道的好。
这里我最后是学习了题解中的经验,用容器存储了每一个节点的代表节点的出现次数,以这个作为统计并查集中节点数的标准确实是最好的,这个可能是一个普遍的技巧吧,不过对于我这个只对于并查集有一定概念了解的新人来说,还是很巧妙有帮助的。
不管咋说,做一个困难的题目,还是挺happy的,毕竟之前没少被折磨。
代码
class Solution {
public:
vector<int> pre; //存储每个结点的前驱结点
vector<int> rank; //树的高度
void init(int n) //初始化函数,对录入的 n个结点进行初始化
{
for(int i = 0; i < n; i++){
pre.push_back(i); //每个结点的上级都是自己
rank.push_back(1); //每个结点构成的树的高度为 1
}
}
int find(int x)
{
if(pre[x] == x){
return x;
}
pre[x] = find(pre[x]);
return pre[x];
}
bool isSame(int x, int y) //判断两个结点是否连通
{
return find(x) == find(y); //判断两个结点的根结点(即代表元)是否相同
}
bool join(int x,int y)
{
x = find(x); //寻找 x的代表元
y = find(y); //寻找 y的代表元
if(x == y) return false; //如果 x和 y的代表元一致
if(rank[x] > rank[y]) pre[y]=x;
else
{
if(rank[x]==rank[y]) rank[y]++;
pre[x]=y;
}
// pre[y]=x;
// rank[x]=max(rank[x],rank[y]+1);
// cout<<x<<" "<<y<<" "<<pre[y]<<" "<<rank[x]<<endl;
return true; //返回 true,表示合并成功
}
int largestComponentSize(vector<int>& nums) {
int l=nums.size();
int max=*max_element(nums.begin(),nums.end());
init(max+1);
for(int i=0;i<l;i++){
int shu=nums[i];
// if(shu==2){
// join(shu,1);
// continue;
// }
for(int j=2;j*j<=shu;j++){
if(shu%j==0){
join(shu,j);
join(shu,shu/j);
}
}
}
int ans=0;
vector<int> count(max+1);
for(int i=0;i<l;i++){
cout<<i<<" "<<rank[nums[i]]<<endl;
int tou=find(nums[i]);
count[tou]++;
ans=ans>count[tou]?ans:count[tou];
}
return ans;
}
};
边栏推荐
- 千万级别的表分页查询非常慢,怎么办?
- 解决报错: YarnScheduler: Initial job has not accepted any resources
- 金仓数据库 KDTS 迁移工具使用指南 (5. SHELL版使用说明)
- Detailed ResNet: What problem is ResNet solving?
- a标签下载图片,不要预览
- LeetCode 97. 交错字符串
- 千古第一文人苏轼的众CP
- 【JS 逆向百例】某网站加速乐 Cookie 混淆逆向详解
- 一天学会JDBC04:ResultSet的用法
- Distributed Computing Experiment 1 Load Balancing
猜你喜欢
一天学会JDBC04:ResultSet的用法
MySQL BIGINT 数据类型
设计信息录入界面,完成人员基本信息的录入工作,
串口监听 - 软件方案
Amazon亚马逊 Vendor Central Label详解
玩转TypeScript对象、对象作为参数进行函数传递、接口和内置对象[无敌态]
第一次用postgreSQL,想装主从,用的12.7 tar.gz版本。安装好后没在 share目录下找到样例配置recovery.conf.sample,是安装方式不对,还是路径不对?
[Paper Notes] - Low Illumination Image Enhancement - Supervised - RetinexNet - 2018-BMVC
Lightweight Backbone VGNetG Achieves "No Choice, All" Lightweight Backbone Network
开发小技巧 navicate如何点击单元格显示全部的文本内容或通过图像查看内容
随机推荐
关于常用状态码4XX提示错误
ConstraintSet of animation of ContrstrainLayout
使用腾讯云发送短信 ---- 手把手教你搞定所有步骤
在GBase 8c数据库后台,使用什么样的命令来对gtm、dn节点进行主备切换的操作?
app逆向1某联
Distributed Computing Experiment 3 PRC-based Book Information Management System
打破千篇一律,DIY属于自己独一无二的商城
The national vocational skills contest competition of network security emergency response
卷积神经网络CNN
国内外知名源码商城系统盘点
在线问题反馈模块实战(十八):实现excel台账文件记录批量导入功能
创建数据库报错--MySQL server is running with the --super-read-only option
1161. Maximum Level Sum of a Binary Tree
小程序如何使用订阅消息(PHP代码+小程序js代码)
为什么手动启动GBase 8c数据库中GTM节点,起不来。显示“Run cmd failed:scp: /tmp/gtm_gtm1.server: Permission denied”
leetcode 22.7.31(1)两数之和 (2)整数除法
错误记录:TypeError: object() takes no parameters
分布式计算实验1 负载均衡
分布式计算实验2 线程池
第一次用postgreSQL,想装主从,用的12.7 tar.gz版本。安装好后没在 share目录下找到样例配置recovery.conf.sample,是安装方式不对,还是路径不对?