Leetcode(540)——有序数组中的单一元素

本文介绍了一种在有序数组中高效查找唯一出现一次元素的方法,采用二分查找算法实现O(log n)的时间复杂度。

Leetcode(540)——有序数组中的单一元素

题目

给你一个仅由整数组成的有序数组,其中每个元素都会出现两次,唯有一个数只会出现一次。

请你找出并返回只出现一次的那个数。

你设计的解决方案必须满足 O(log⁡n)O(\log n)O(logn) 时间复杂度和 O(1)O(1)O(1) 空间复杂度。

示例 1:

输入: nums = [1,1,2,3,3,4,4,8,8]
输出: 2

示例 2:

输入: nums = [3,3,7,7,10,11,11]
输出: 10

提示:

  • 111 <= nums.length <= 10510^5105
  • 000 <= nums[i] <= 10510^5105

题解

方法一:全数组的二分查找

思路

​​  由于给定数组有序 且 常规元素总是两两出现,因此如果不考虑“特殊”的单一元素的话,我们有结论:成对元素中的第一个所对应的下标必然是偶数,成对元素中的第二个所对应的下标必然是奇数
​​  然后再考虑存在单一元素的情况,假如单一元素所在的下标为 xxx,那么下标 xxx 之前(左边)的位置仍满足上述结论,而下标 xxx 之后(右边)的位置由于 xxx 的插入,导致结论翻转

​​  假设只出现一次的元素位于下标 xxx,由于其余每个元素都出现两次,因此下标 xxx 的左边和右边都有偶数个元素,数组的长度是奇数。

​​  由于数组是有序的,因此数组中相同的元素一定相邻。对于下标 xxx 左边的下标 yyy,如果 nums[y]=nums[y+1]\textit{nums}[y] = \textit{nums}[y + 1]nums[y]=nums[y+1],则 yyy 一定是偶数;对于下标 xxx 右边的下标 zzz,如果 nums[z]=nums[z+1]\textit{nums}[z] = \textit{nums}[z + 1]nums[z]=nums[z+1],则 zzz 一定是奇数。由于下标 xxx 是相同元素的开始下标的奇偶性的分界,因此可以使用二分查找的方法寻找下标 xxx

初始时,二分查找的左边界是 000,右边界是数组的最大下标。每次取左右边界的平均值 mid\textit{mid}mid 作为待判断的下标,根据 mid\textit{mid}mid 的奇偶性决定和左边或右边的相邻元素比较:

  • 如果 mid\textit{mid}mid 是偶数,则比较 nums[mid]\textit{nums}[\textit{mid}]nums[mid]nums[mid+1]\textit{nums}[\textit{mid} + 1]nums[mid+1] 是否相等;
  • 如果 mid\textit{mid}mid 是奇数,则比较 nums[mid−1]\textit{nums}[\textit{mid} - 1]nums[mid1]nums[mid]\textit{nums}[\textit{mid}]nums[mid] 是否相等。

如果上述比较相邻元素的结果是相等,则 mid<x\textit{mid} < xmid<x,调整左边界,否则 mid≥x\textit{mid} \ge xmidx,调整右边界。调整边界之后继续二分查找,直到确定下标 xxx 的值。得到下标 xxx 的值之后,nums[x]\textit{nums}[x]nums[x] 即为只出现一次的元素。

细节(小技巧)——不需要判断 mid\textit{mid}mid 的奇偶性

利用按位异或的性质,可以得到 mid\textit{mid}mid 和相邻的数之间的如下关系,其中 ⊕\oplus 是按位异或运算符:

  • mid\textit{mid}mid 是偶数时,mid+1=mid⊕1\textit{mid} + 1 = \textit{mid} \oplus 1mid+1=mid1
  • mid\textit{mid}mid 是奇数时,mid−1=mid⊕1\textit{mid} - 1 = \textit{mid} \oplus 1mid1=mid1

因此在二分查找的过程中,不需要判断 mid\textit{mid}mid 的奇偶性,mid\textit{mid}midmid⊕1\textit{mid} \oplus 1mid1 即为每次需要比较元素的两个下标

因为 0⊕10 \oplus 101 的结果为 111 ,所以不用担心左侧会越界。又因为 mid=(l+r)/2mid = (l+r)/2mid=(l+r)/2 所以 midmidmid 的值是向下取整,不会出现右侧访问数组越界。

代码实现

Leetcode 官方题解:

class Solution {
public:
    int singleNonDuplicate(vector<int>& nums) {
        int l = 0, r = nums.size()-1, mid;
        while(l < r){
            mid = (l+r)/2;
            if(nums[mid] == nums[mid^1]) 
            	l = mid + 1;
            else r = mid;
        }
        return nums[l];
    }
};

我自己的(用长度来判定):

class Solution {
public:
    int singleNonDuplicate(vector<int>& nums) {
        // 找到 mid 判断其左右区间的长度,找到长度为奇数的继续查找下去
        // 比如 [1,1,2,3,3,4,4] mid 为第一个3,然后判断它和与它相同的数值的右边区间的长度是否为奇数,是则查找,不是则查找另一个
        int l = 0, r = nums.size()-1, mid, n = nums.size();
        while(l <= r){
            mid = (l+r)/2;
            if(mid != 0 && nums[mid] == nums[mid-1]){
                if((r-mid)%2 == 0) r = mid-2;
                else l = mid+1;
            }else if(mid != n-1 && nums[mid] == nums[mid+1]){
                if((r-mid+1)%2 == 0) r = mid-1;
                else l = mid+2;
            }else break;
        }
        return nums[mid];
    }
};
复杂度分析

时间复杂度O(logn)O(logn)O(logn),其中 nnn 是数组 nums\textit{nums}nums 的长度。需要在全数组范围内二分查找,二分查找的时间复杂度是 O(log⁡n)O(\log n)O(logn)
空间复杂度O(1)O(1)O(1)

方法二:偶数下标的二分查找

思路

​​  由于只出现一次的元素所在下标 xxx 的左边有偶数个元素,因此下标 xxx 一定是偶数,可以在偶数下标范围内二分查找。二分查找的目标是找到满足 nums[x]≠nums[x+1]\textit{nums}[x] \ne \textit{nums}[x + 1]nums[x]=nums[x+1] 的最小的偶数下标 xxx,则下标 xxx 处的元素是只出现一次的元素

​​  初始时,二分查找的左边界是 000,右边界是数组的最大偶数下标,由于数组的长度是奇数,因此数组的最大偶数下标等于数组的长度减 111
​​  每次取左右边界的平均值 mid\textit{mid}mid 作为待判断的下标,如果 mid\textit{mid}mid 是奇数则将 mid\textit{mid}mid111,确保 mid\textit{mid}mid 是偶数。
​​  比较 nums[mid]\textit{nums}[\textit{mid}]nums[mid]nums[mid+1]\textit{nums}[\textit{mid} + 1]nums[mid+1] 是否相等,如果相等则 mid<x\textit{mid} < xmid<x,调整左边界,否则 mid≥x\textit{mid} \ge xmidx,调整右边界。调整边界之后继续二分查找,直到确定下标 xxx 的值。
​​  得到下标 xxx 的值之后,nums[x]\textit{nums}[x]nums[x] 即为只出现一次的元素。

细节

考虑 mid\textit{mid}mid111 按位与运算的结果,其中 &\&& 是按位与运算符

  • mid\textit{mid}mid 是偶数时,mid & 1=0\textit{mid}~\&~1 = 0mid & 1=0
  • mid\textit{mid}mid 是奇数时,mid & 1=1\textit{mid}~\&~1 = 1mid & 1=1

因此在得到 mid\textit{mid}mid 的值之后,mid\textit{mid}mid 的值减去 mid & 1\textit{mid}~\&~1mid & 1,即可确保新得到的 mid\textit{mid}mid 是偶数。如果原来的 mid\textit{mid}mid 是偶数则值不变,如果原来的 mid\textit{mid}mid 是奇数则值减 111

代码实现

Leetcode 官方题解:

class Solution {
public:
    int singleNonDuplicate(vector<int>& nums) {
        int low = 0, high = nums.size() - 1;
        while (low < high) {
            int mid = (high - low) / 2 + low;
            mid -= mid & 1;
            if (nums[mid] == nums[mid + 1]) {
                low = mid + 2;
            } else {
                high = mid;
            }
        }
        return nums[low];
    }
};
复杂度分析

时间复杂度O(log⁡n)O(\log n)O(logn),其中 nnn 是数组 nums\textit{nums}nums 的长度。需要在偶数下标范围内二分查找,二分查找的时间复杂度是 O(log⁡n)O(\log n)O(logn)
空间复杂度O(1)O(1)O(1)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值