有序数组中的单一元素

题目

在这里插入图片描述

解题思路

由于只有一个数只出现一次,其他数均出现两次,而且数组是有序的。
因此在只出现一次的数之前,偶数位的数一定与后面相邻的奇数位的数相同。
在该数之后,奇数位的数与后面相邻的偶数位的数相同
(例如数组[1,1,2,3,3,4,4,8,8]中,序号2之前,nums[0] == nums[1],序号2之后,nums[3] == nums[4]等等)

因此判断条件就是偶数位与后面相邻的奇数位是否相同,
若相同,则表明要求的答案在当前结果的后面,
否则,表明答案不在当前结果后面。

给定一个序号p,求该序号所在的 偶数位-奇数位 的位置时,可以通过 (p/2)*2 和 (p/2)*2 + 1求取,少了一个判断。

Code

class Solution {
    public int singleNonDuplicate(int[] nums) {
       int l=0;
        int h=nums.length-1;
        if (h==0){
            return nums[0];
        }
        while (l<h){
            int m=(l+h)/2;
            int side=m-l+1;//一侧连中间的轴共有多少数,
            if(nums[m+1]!=nums[m]&&nums[m-1]!=nums[m]){
                return nums[m];
            }
            if(nums[m-1]==nums[m]){
                if(side%2==0){
                   l=m+1;
                }
                else {
                    h=h-2;
                }
            }
            else {
                if(side%2==0){
                    h=m-1;
                }
                else {
                    l=l+2;
                }
            }

        }
        return nums[l];
    }
}

代码运行截图
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值