力扣5866. 数组的最大公因数排序

原题:5866. 数组的最大公因数排序
**1.题意为任意两个数的公因数大于1,那么这两个数可以交换。由此可知如果a和b可交换,b和c可交换,那么a,b,c三者可以任意改变顺序,不难想到用并查集把所有公因数大于1的两个数合并。

2.如果用两层循环来判断来合并任意两个数,此时必然会超时。因此考虑将每个数和自己的所有质因子进行合并,如15和质因子3,5进行合并,21和质因子3,7合并,这样保证了21和15在同一个集合中。这样对于每个数仅仅需要分解质因子的时间复杂度,远远低于两层循环所需的时间复杂度。注:为了降低运行时间,我们只需要存储共同的因子就足够了

3.合并之后,所有在一个并查集中的数可以任意交换。将原有的数组进行排序和新数组对比,如果原有数组和新数组的数字相同,则跳过,如果不同,则必须满足两个数在同一个并查集中,否则返回false,扫描一遍后如果没有返回false,则返回true.**

#define N 100005
int fathers[N];
// 求最大公因数
// 以后手写了,有个好东西!
//__gcd(x, y)
class Solution {
public:
    int find(int cur) {
        if (fathers[cur] == cur) return cur;
        return (fathers[cur] = find(fathers[cur]));
    }

    void merge(int a, int b) {
        int fa = find(a);
        int fb = find(b);
        fathers[fa] = fb;
    }

    bool gcdSort(vector<int> &nums) {
        for (int i = 0; i < N; i++) fathers[i] = i;
        //最少量地存储,比如24存:2;  22存:2、11  
        for(int num: nums){
            int temp = num;
            for(int i=2; i*i<=temp; i++){
                if(temp%i!=0)   continue;
                while(temp%i==0)    temp /= i;
                merge(num, i);
            }
            if(temp>1)  merge(num, temp);
        }
        // 最原始的求所有因素的做法,比较慢,不过还是能过
        for(int num: nums){
            for(int i=2; i<=sqrt(num); i++){
                if(num%i==0)    {
                    merge(num, i);
                    merge(num, num/i);
                }
            }
        }
        vector<int> copy = nums;
        sort(copy.begin(), copy.end(), less<int>());
        for(int i=0; i<nums.size(); i++){
            if(copy[i]==nums[i])    continue;
            // printf("%d %d %d %d\n", copy[i], nums[i], find(copy[i]), find(nums[i]));
            if(find(copy[i])!=find(nums[i]))    return false;
        }
        return true;
    }
};




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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值