LeetCode[765]情侣牵手

难度:困难

题目:

n 对情侣坐在连续排列的 2n 个座位上,想要牵到对方的手。
人和座位由一个整数数组 row 表示,其中 row[i] 是坐在第 i 个座位上的人的 ID。情侣们按顺序编号,第一对是 (0, 1) ,第二对是 (2, 3) ,以此类推,最后一对是 (2n-2, 2n-1)
返回 最少交换座位的次数,以便每对情侣可以并肩坐在一起每次交换可选择任意两人,让他们站起来交换座位。

示例 1:

输入: row = [0,2,1,3]

输出: 1

解释: 只需要交换row[1]和row[2]的位置即可。


示例 2:

输入: row = [3,2,0,1]

输出: 0

解释: 无需交换座位,所有的情侣都已经可以手牵手了。


提示:

  • 2n == row.length

  • 2 <= n <= 30

  • n 是偶数

  • 0 <= row[i] < 2n

  • row 中所有元素均无重复


Related Topics

  • 贪心

  • 深度优先搜索

  • 广度优先搜索

  • 并查集


重点!!!解题思路

第一步:

明确解题手段
类似于此种连通性问题均能使用并查集来实现

第二步:

可以看题和例子找出规律,每个人都是0 1 2 3...n的
每个情侣之前除以2都可以得到这个情侣的下标
比如0 1都除以2,第0个情侣下标
比如2 3都除以2,第一个情侣下标
找出规律后:
我们使用并查集将每次遍历的两个数除以2,
如果相同就不连接,如果不相同就连接

第三步:

总数-连通分量=操作次数
如果一次都没有操作成功,那么连通分量=总数,即返回0

源码:

class UnionFind {
    //记录每个节点的根节点
    int[] parent;
    //记录每个子集的节点数
    int[] rank;
    //记录并查集中的联通分量数量
    int count;

    public UnionFind(int n){
        count=n;
        parent=new int[n];
        for (int i=0;i>1,row[i+1]>>1);//连接的时候如果是同一个节点,或在同一个集合内就不连接
        }
        return N-uf.getCount();
    }
}

运行结果:

LeetCode[765]情侣牵手_第1张图片

如果您还有什么疑问或解答有问题,可在下方评论,我会及时回复。

系列持续更新中,点个订阅吧

你可能感兴趣的:(算法刷题篇,#,树结构基础(刷题篇),leetcode,算法,数据结构,java)