【2023】字节跳动 10 日心动计划——第六关

目录

  • [1. 环形链表 II](#1. 环形链表 II)
  • [2. 有序数组中的单一元素](#2. 有序数组中的单一元素)

1. 环形链表 II

🔗 原题链接:142. 环形链表 II

用哈希表判重即可。

cpp 复制代码
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        unordered_set<ListNode*> st;
        while (head) {
            if (st.count(head)) return head;
            st.insert(head);
            head = head->next;
        }
        return nullptr;
    }
};

2. 有序数组中的单一元素

🔗 原题链接:LCR 070. 有序数组中的单一元素

这里介绍两种做法。

方法一:异或。注意到 x ⊕ x = 0 x\oplus x=0 x⊕x=0,因此 ⊕ i = 1 n n u m s i \oplus_{i=1}^nnumsi ⊕i=1nnumsi 就是最终答案。

cpp 复制代码
class Solution {
public:
    int singleNonDuplicate(vector<int>& nums) {
        int ans = 0;
        for (auto& num: nums) ans ^= num;
        return ans;
    }
};

方法二:二分。注意到数组是有序的,因此我们可以用二分去处理,时间复杂度可以降至 O ( log ⁡ n ) O(\log n) O(logn)。

不妨设只出现一次的元素的下标是 x x x,由题意可知下标 x x x 的左右两边各有偶数个元素,从而 x x x 是偶数,且数组的长度是奇数,不妨设为 n n n。

既然要用二分,那我们就需要找到一个性质能够将区间 0 , n − 1 0,n-1 0,n−1 一分为二。注意到 ∀ i ∈ 0 , x − 1 \forall i\in0,x-1 ∀i∈0,x−1,如果 n u m s i = n u m s i + 1 numsi=numsi+1 numsi=numsi+1,那么 i i i 一定是偶数; ∀ i ∈ x , n − 1 \forall i\inx,n-1 ∀i∈x,n−1,如果 n u m s i = n u m s i + 1 numsi=numsi+1 numsi=numsi+1,那么 i i i 一定是奇数。

更进一步, ∀ i ∈ 0 , n − 1 \forall i\in0,n-1 ∀i∈0,n−1,如果 i i i 是偶数,我们判断 n u m s i numsi numsi 和 n u m s i + 1 numsi+1 numsi+1 是否相等,如果相等,则 i ∈ 0 , x − 1 i\in0,x-1 i∈0,x−1,否则 i ∈ x , n − 1 i\inx,n-1 i∈x,n−1;如果 i i i 是奇数,则 i − 1 i-1 i−1 是偶数,我们判断 n u m s i − 1 numsi-1 numsi−1 和 n u m s i numsi numsi 是否相等,如果相等,则 i ∈ 0 , x − 1 i\in0,x-1 i∈0,x−1,否则 i ∈ x , n − 1 i\inx,n-1 i∈x,n−1

注意到当 i i i 是偶数时, i + 1 = i ⊕ 1 i+1=i\oplus1 i+1=i⊕1,当 i i i 是奇数时, i − 1 = i ⊕ 1 i-1=i\oplus1 i−1=i⊕1,从而我们只需要判断 n u m s i numsi numsi 和 n u m s i ⊕ 1 numsi\\oplus 1 numsi⊕1 是否相等。条件 n u m s i ≠ n u m s i ⊕ 1 numsi\neq numsi\\oplus1 numsi=numsi⊕1 将区间 0 , n − 1 0,n-1 0,n−1 分成了两部分: 0 , x ) \[0,x) \[0,x) 和 \[ x , n − 1 x,n-1 x,n−1,前者不满足这个条件,后者满足这个条件,所以我们可以套用寻找左边界的二分模版

cpp 复制代码
class Solution {
public:
    int singleNonDuplicate(vector<int>& nums) {
        int l = 0, r = nums.size() - 1;
        while (l < r) {
            int mid = l + r >> 1;
            if (nums[mid] != nums[mid ^ 1]) r = mid;
            else l = mid + 1;
        }
        return nums[r];
    }
};
相关推荐
罗西的思考8 小时前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
CSharp精选营10 小时前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型
美团技术团队11 小时前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法
To_OC1 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC1 天前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
BadBadBad__AK1 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
_清歌2 天前
DSpark 深度解读:DeepSeek-V4 如何用「半自回归」把推理速度提升 85%
算法
统计实现局2 天前
SVD 的三步走:双对角化、Givens 收敛、排序
算法
躬行见万象2 天前
《VLA 系列》UniLab 强化训练 | G1 机器人 |复现
算法