被面试官发现自己是海王

尴尬

今天在学生集中地看到了一篇极其吸睛的文章:

alt

这位同学表示自己在参加快手三面时,共享屏幕时无意暴露了自己的文档,里面记录了自己的秋招进度,虽然自己迅速切换了界面,但还是被面试官看到了,还被面试官当面感叹。

帖子底部,这些同学分享自己在目前秋招的进度:字节和腾讯均已 OC,其他大厂也有二面、三面甚至 HR 面的进度。

好家伙,我直呼好家伙,这简直是秋招领域的"炫富"。

手上都这么多进度了,还用担心这一两个面试官发现了怎么看?不如担心一下评论区的 0Offer 选手看到怎么看 🤣🤣🤣

alt
alt

不少同学还是 0Offer 甚至是 0 约面的状态,这一下子让大伙破防了,纷纷表示给"给大伙留点机会吧"。

怎么说呢,虽然相比几年前,目前的校招已卷出新高度,但优秀的人才还是十分稀缺,在绝对实力的前面,行情也就图一乐。

alt

评论区有同学感叹道,同样都是秋招,对大佬来说拿 Offer 就像集邮一样简单,对自己而言却是九九八十一难。

对此,你怎么看?你是否在工作或面试中有过类似尴尬时刻?

...

回归主题。

来一道和「某厂秋招」相关的算法原题。

题目描述

平台:LeetCode

题号:765

N 对情侣坐在连续排列的 2N 个座位上,想要牵到对方的手。

计算最少交换座位的次数,以便每对情侣可以并肩坐在一起。 一次交换可选择任意两人,让他们站起来交换座位。

人和座位用 0 到 2N-1 的整数表示,情侣们按顺序编号,第一对是  ,第二对是  ,以此类推,最后一对是 

这些情侣的初始座位  row[i] 是由最初始坐在第 i 个座位上的人决定的。

示例 1:

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

输出: 1

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

示例 2:

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

输出: 0

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

说明:

  • len(row) 是偶数且数值在  范围内。
  • 可以保证 row 是序列  0...len(row)-1 的一个全排列。

并查集

首先,我们总是以「情侣对」为单位进行设想:

  1. 当有两对情侣相互坐错了位置,ta 们两对之间形成了一个环。需要进行一次交换,使得每队情侣独立(相互牵手)

  2. 如果三对情侣相互坐错了位置,ta 们三对之间形成了一个环,需要进行两次交换,使得每队情侣独立(相互牵手)

  3. 如果四对情侣相互坐错了位置,ta 们四对之间形成了一个环,需要进行三次交换,使得每队情侣独立(相互牵手)

也就是说,如果我们有 k 对情侣形成了错误环,需要交换 k - 1 次才能让情侣牵手。

「于是问题转化成 n / 2 对情侣中,有多少个这样的环。」

可以直接使用「并查集」来做。

由于 0和1配对、2和3配对 ... 因此互为情侣的两个编号除以 2 对应同一个数字,可直接作为它们的「情侣组」编号:

Java 代码:

 class Solution {
    int[] p = new int[70];
    void union(int a, int b) {
        p[find(a)] = p[find(b)];
    }
    int find(int x) {
        if (p[x] != x) p[x] = find(p[x]);
        return p[x];
    }
    public int minSwapsCouples(int[] row) {
        int n = row.length, m = n / 2;
        for (int i = 0; i < m; i++) p[i] = i;
        for (int i = 0; i < n; i += 2) union(row[i] / 2, row[i + 1] / 2);
        int cnt = 0;
        for (int i = 0; i < m; i++) {
            if (i == find(i)) cnt++;
        }
        return m - cnt;
    }
}

C++ 代码:

class Solution {
public:
    vector<int> p;
    Solution() : p(70) {
        for (int i = 0; i < 70; ++i) p[i] = i;
    }
    void unionSet(int a, int b) {
        p[find(a)] = p[find(b)];
    }
    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(), m = n / 2;
        for (int i = 0; i < m; i++) p[i] = i;
        for (int i = 0; i < n; i += 2) unionSet(row[i] / 2, row[i + 1] / 2);
        int cnt = 0;
        for (int i = 0; i < m; i++) {
            if (i == find(i)) ++cnt;
        }
        return m - cnt;
    }
};

Python 代码:

class Solution:
    def __init__(self):
        self.p = [i for i in range(70)]
        
    def union(self, a, b):
        self.p[self.find(a)] = self.p[self.find(b)]
        
    def find(self, x):
        if self.p[x] != x:
            self.p[x] = self.find(self.p[x])
        return self.p[x]
    
    def minSwapsCouples(self, row):
        n = len(row)
        m = n // 2
        for i in range(0, n, 2):
            self.union(row[i] // 2, row[i + 1] // 2)
        cnt = 0
        for i in range(m):
            if i == self.find(i):
                cnt += 1
        return m - cnt
  • 时间复杂度:
  • 空间复杂度:

贪心

还是以「情侣对」为单位进行分析:

由于题目保证有解,我们也可以从前往后(每两格作为一步)处理,对于某一个位置而言,如果下一个位置不是应该出现的情侣的话。

则对下一个位置进行交换。

同时为了方便我们找到某个值的下标,需要先对 row 进行预处理(可以使用哈希表或数组)。

Java 代码:

class Solution {
    public int minSwapsCouples(int[] row) {
        int n = row.length, ans = 0;
        int[] cache = new int[n];
        for (int i = 0; i < n; i++) cache[row[i]] = i;
        for (int i = 0; i < n - 1; i += 2) {
            int a = row[i], b = a ^ 1;
            if (row[i + 1] != b) {
                int src = i + 1, tar = cache[b];
                cache[row[tar]] = src;
                cache[row[src]] = tar;
                swap(row, src, tar);
                ans++;
            }
        }
        return ans;
    }
    void swap(int[] nums, int a, int b) {
        int c = nums[a];
        nums[a] = nums[b];
        nums[b] = c;
    }
}

C++ 代码:

class Solution {
public:
    int minSwapsCouples(vector<int>& row) {
        int n = row.size(), ans = 0;
        vector<intcache(n);
        for (int i = 0; i < n; i++) cache[row[i]] = i;
        for (int i = 0; i < n - 1; i += 2) {
            int a = row[i], b = a ^ 1;
            if (row[i + 1] != b) {
                int src = i + 1, tar = cache[b];
                cache[row[tar]] = src;
                cache[row[src]] = tar;
                swap(row[src], row[tar]);
                ans++;
            }
        }
        return ans;
    }
};

Python 代码:

class Solution:
    def minSwapsCouples(self, row):
        n, ans = len(row), 0
        cache = [0] * (max(row) + 1)
        for i in range(n):
            cache[row[i]] = i
        for i in range(0, n, 2):
            a = row[i]
            b = a ^ 1
            if row[i + 1] != b:
                src = i + 1
                tar = cache[b]
                cache[row[tar]] = src
                cache[row[src]] = tar
                row[src], row[tar] = row[tar], row[src]
                ans += 1
        return ans
  • 时间复杂度:
  • 空间复杂度:

证明/分析

我们这样的做法本质是什么?

「其实相当于,当我处理到第 k 个位置的时候,前面的 k - 1 个位置的情侣已经牵手成功了。我接下来怎么处理,能够使得总花销最低。」

分两种情况讨论:

a. 现在处理第 k 个位置,使其牵手成功:

那么我要使得第 k 个位置的情侣也牵手成功,那么必然是保留第 k 个位置的情侣中其中一位,再进行修改,这样的成本是最小的(因为只需要交换一次)。

而且由于前面的情侣已经牵手成功了,因此交换的情侣必然在 k 位置的后面。

然后我们再考虑交换左边或者右边对最终结果的影响。

分两种情况来讨论:

  1. 与第 k 个位置的匹配的两个情侣不在同一个位置上:这时候无论交换左边还是右边,后面需要调整的「情侣对数量」都是一样。假设处理第 k 个位置前需要调整的数量为 n 的话,处理完第 k 个位置(交换左边或是右边),需要调整的「情侣对数量」都为 n - 1
alt
  1. 与第 k 个位置的匹配的两个情侣在同一个位置上:这时候无论交换左边还是右边,后面需要调整的「情侣对数量」都是一样。假设处理第 k 个位置前需要调整的数量为 n 的话,处理完第 k 个位置(交换左边或是右边),需要调整的「情侣对数量」都为 n - 2
alt

因此对于第 k 个位置而言,交换左边还是右边,并不会影响后续需要调整的「情侣对数量」。

b. 现在先不处理第 k 个位置,等到后面的情侣处理的时候「顺便」处理第 k 位置:

由于我们最终都是要所有位置的情侣牵手,而且每一个数值对应的情侣数值是唯一确定的。

因此我们这个等“后面”的位置处理,其实就是等与第 k 个位置互为情侣的位置处理(对应上图的就是我们是在等 【0 x】和【8 y】或者【0 8】这些位置被处理)。

由于被处理都是同一批的联通位置,因此和「a. 现在处理第 k 个位置」的分析结果是一样的。

不失一般性的,我们可以将这个分析推广到第一个位置,其实就已经是符合「当我处理到第 k 个位置的时候,前面的 k - 1 个位置的情侣已经牵手成功了」的定义了。

「综上所述,我们只需要确保从前往后处理,并且每次处理都保留第 k 个位置的其中一位,无论保留的左边还是右边都能得到最优解。」

最后

巨划算的 LeetCode 会员优惠通道目前仍可用 ~

使用福利优惠通道 leetcode.cn/premium/?promoChannel=acoier,年度会员 有效期额外增加两个月,季度会员 有效期额外增加两周,更有超大额专属 🧧 和实物 🎁 福利每月发放。

我是宫水三叶,每天都会分享算法知识,并和大家聊聊近期的所见所闻

欢迎关注,明天见。

更多更全更热门的「笔试/面试」相关资料可访问排版精美的 合集新基地 🎉🎉

  • 18
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值