剑指 Offer II 070(力扣540):排序数组中只出现一次的数字(Java二分查找)

简介: 给定一个只包含整数的有序数组 nums ,每个元素都会出现两次,唯有一个数只会出现一次,请找出这个唯一的数字。

一、题目描述



给定一个只包含整数的有序数组nums ,每个元素都会出现两次,唯有一个数只会出现一次,请找出这个唯一的数字。


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


示例 1:

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

输出: 2


示例 2:

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

输出: 10


提示:

  • 1 <= nums.length <= 105
  • 0 <= nums[i] <= 105


二、思路讲解



2.1 哈希表

     

找个数的题目,用HashMap可以通杀。使用哈希表保存每个数字出现的次数,然后找到值为1的即可。


2.2 一次循环


因为数组本身是有序的(即使的无序的也可以给他排序),所以相等的数字必然相邻,那么我们可以一遍循环,找到一个数字,他的左边和右边跟自己都不相等,这个数字就是只出现一次的数字。


时间复杂度:        O(N)

空间复杂度:        O(1)

       

2.3 二分查找


然而,题目中要求的时间复杂度为对数级。由经验可得,有序数组的查找基本上可以使用二分查找,因此我们取左指针low指向最左,右指针high指向最右,中点mid指向中间,可以注意到:  

     

1、当mid为偶数时:

——如果nums[mid]跟左边的数字相等,那么所求数字一定出现在mid的左边,可以直接令high = mid -1。例如:


                               1        1        2        3        3        4        4        5        5


                               low                                 mid                                   high


       ——如果nums[mid]跟右边数字相等,那么所求数字一定出现在mid右边。例如:


                               1        1        2        2        3        3        4        5        5


2、当mid为奇数时:


       ——如果nums[mid]跟左边数字相等,那么所求数字定出现在mid右边。例如:


                     1        1        2        2        3        3        4        4        5        6        6


                   low                                              mid                                            high


       ——如果nums[mid]跟右边数字相等,则相反。例如:


                     1        1        2        3        3        4        4        5        5        6        6


       这样我们就已经确定了二分查找的大致思路。


剩下的困难就是边界的判断问题了,我们一起看看代码里是怎么做的:

class Solution {
    public int singleNonDuplicate(int[] nums) {
        if(nums.length==1 || nums[0]!=nums[1]) {
            return nums[0];
        }
        int low = 0;
        int high = nums.length - 1;
        while(low <= high) {
            //这样写在任何情况下都不会溢出
            int mid = (high - low) / 2 + low; 
            //如果mid达到了边界,就给left或right一个不可能的值,这样可以简化边界的判断
            int left = (mid>0)? nums[mid-1] : -1;
            int right = (mid<nums.length-1)? nums[mid+1] : Integer.MAX_VALUE;
            //如果nums[mid]跟两边都不相等,说明我们找的就是他
            if(nums[mid]!=left && nums[mid]!=right) {
                return nums[mid];
            }
            if(mid%2 == 0) {
                if(nums[mid] == right) {
                    low = mid + 1;
                } else{
                    high = mid -1;
                }
            } else {
                if(nums[mid] == left) {
                    low = mid + 1;
                } else {
                    high = mid - 1;
                }
            }
        }
        return nums[low];
    }
}


时间复杂度:        O(logN)

空间复杂度:        O(1)


2.4 位运算


利用按位异或的性质,可以减少对mid判断奇偶性的步骤,减少代码量,算得上是奇技淫巧了:


当mid为偶数时,mid + 1 = mid ^ 1;

当mid为奇数时,mid  - 1 = mid ^ 1;

class Solution {
    public int singleNonDuplicate(int[] nums) {
        int low = 0, high = nums.length - 1;
        while (low < high) {
            int mid = (high - low) / 2 + low;
            if (nums[mid] == nums[mid ^ 1]) {
                low = mid + 1;
            } else {
                high = mid;
            }
        }
        return nums[low];
    }
}
作者:力扣官方题解
链接:https://leetcode.cn/problems/skFtm2/solutions/1252765/pai-xu-shu-zu-zhong-zhi-chu-xian-yi-ci-d-jk8w/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


目录
打赏
0
0
0
0
178
分享
相关文章
Java 中数组的多种定义方式
本文深入解析了Java中数组的多种定义方式,涵盖基础的`new`关键字创建、直接初始化、动态初始化,到多维数组、`Arrays.fill()`方法以及集合类转换为数组等高级用法。通过理论与实践结合的方式,探讨了每种定义方法的适用场景、优缺点及其背后的原理,帮助开发者掌握高效、灵活的数组操作技巧,从而编写更优质的Java代码。
37 0
Java 中数组Array和列表List的转换
本文介绍了数组与列表之间的相互转换方法,主要包括三部分:1)使用`Collections.addAll()`方法将数组转为列表,适用于引用类型,效率较高;2)通过`new ArrayList&lt;&gt;()`构造器结合`Arrays.asList()`实现类似功能;3)利用JDK8的`Stream`流式计算,支持基本数据类型数组的转换。此外,还详细讲解了列表转数组的方法,如借助`Stream`实现不同类型数组间的转换,并附带代码示例与执行结果,帮助读者深入理解两种数据结构的互转技巧。
Java 中数组Array和列表List的转换
Java 复制数组
本文介绍了Java中数组的基础知识与常用操作,包括数组的概念、创建、访问元素、遍历、复制、排序和搜索等方法。同时详细讲解了数组的五种赋值方式,并通过代码示例演示了求总和平均值、最大最小值、升序降序排序及Arrays类的常用方法。内容深入浅出,适合初学者学习掌握Java数组的核心功能与应用场景。
|
1月前
|
《从头开始学java,一天一个知识点》之:数组入门:一维数组的定义与遍历
**你是否也经历过这些崩溃瞬间?** - 看了三天教程,连`i++`和`++i`的区别都说不清 - 面试时被追问&quot;`a==b`和`equals()`的区别&quot;,大脑突然空白 - 写出的代码总是莫名报NPE,却不知道问题出在哪个运算符 这个系列就是为你打造的Java「速效救心丸」!我们承诺:每天1分钟,地铁通勤、午休间隙即可完成学习;直击痛点,只讲高频考点和实际开发中的「坑位」;拒绝臃肿,没有冗长概念堆砌,每篇都有可运行的代码标本。明日预告:《多维数组与常见操作》。 通过实例讲解数组的核心认知、趣味场景应用、企业级开发规范及优化技巧,帮助你快速掌握Java数组的精髓。
67 23
|
14天前
|
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
本文涉及InheritableThreadLocal和TTL,从源码的角度,分别分析它们是怎么实现父子线程传递的。建议先了解ThreadLocal。
51 4
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
86 23
|
1月前
|
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
153 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等