765. 情侣牵手(并查集代码)

这道题使用并查集解决。

01 23 45 三对情侣可以看成 00 11 22 三对情侣,在这道题中,可以把0 1 2看作三个点,沙发看作三条边,可以理解成一个环状图,定下其点和边,开始思考。

假设现在有4对情侣,目前的座位是12 23 34 41,将其作为一个图,可以发现这是一个环状图,

如果交换环内两点,例如交换3和1,即这两点12 23 34 41 ,交换后的座位为12 21 34 43 ,可以发现变成了两个环,于是得出结论一:交换环内的点会增加一个环

如果继续交换两个环内的两个点,即两点不属于同一个环,例如在12 21 34 4 3 的基础上交换2和4,得到交换后的座位为14 21 34 2 3,可以发现又变回了一个环,于是得出结论二:交换不同环的两个点会减少一个环

在这道题中,情侣成对就坐相当于两个节点成环 ,一共row.size()个节点,需要那n=row.size()/2个环,将每对情侣视为一体,座位构成的图为无向图,在无向图中,环的数量相当于连通块的数量 ,于是考虑使用并查集来解决问题。

并查集

参考博客并查集

使用数组和双亲表示法,p[x]储存x的双亲节点
并查集代码

cpp 复制代码
int find(int x) {
	if (p[x] != x) p[x] = find(p[x]);
    return p[x];
}

于是题目转换为,目前要求生成n个连通块,目前有cnt个连通块,要求交换多少次才能达到目标,根据前文结论一可以知道需要交换n-cnt次可以生成n个连通块(环),于是n-cnt为最终答案。

cpp 复制代码
class Solution {
public:
    vector<int> p;

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

    int minSwapsCouples(vector<int>& row) {
        int n = row.size() / 2;
        int cnt = n;
        for (int i = 0; i < n; i++) p.push_back(i);
        for (int i = 0; i < n * 2; i += 2) {	// 注意这里i进行步长为2的迭代
            int a = row[i] / 2, b = row[i + 1] / 2;
            if (find(a) != find(b)) {
                p[find(a)] = find(b);
                cnt--;
            }
        }
        return n - cnt;
    }
};
相关推荐
承渊政道2 分钟前
【贪心算法】(经典实战应用解析(六):整数替换、俄罗斯套娃信封问题、可被三整除的最⼤和、距离相等的条形码、重构字符串)
c++·算法·leetcode·贪心算法·排序算法·动态规划·哈希算法
WL_Aurora5 分钟前
Python 算法基础篇之排序算法(二):希尔、快速、归并
python·算法·排序算法
闻缺陷则喜何志丹10 分钟前
【图论 树 启发式合并】P7165 [COCI2020-2021#1] Papričice|普及+
c++·算法·启发式算法·图论··洛谷
alexwang21112 分钟前
AT_abc458_d [ABC458D] Chalkboard Median题解
c++·算法·题解·atcoder
故事和你9112 分钟前
洛谷-【图论2-4】连通性问题1
开发语言·数据结构·c++·算法·动态规划·图论
周末也要写八哥22 分钟前
算法实例分析:使数组相等的最小开销
算法
吃好睡好便好25 分钟前
在Matlab中绘制质点运动轨迹图
开发语言·学习·算法·matlab·信息可视化
爱炼丹的James28 分钟前
第三章 搜索和图论
数据结构·算法·图论
菜菜笔记29 分钟前
【无标题】
算法
努力努力再努力wz34 分钟前
【QT入门系列】QWidget 六大常用属性详解:windowOpacity、cursor、font、focus、toolTip 与 styleSheet
android·开发语言·数据结构·c++·qt·mysql·算法