DAY-7 | 牛客-BM21 寻找旋转数组的最小元素:二分法分治思想真的很可靠

简介: 这是一个关于编程题目的摘要:题目是牛客网上的"BM21 旋转数组的最小数字",要求找到旋转数组中的最小数字。题解介绍了使用二分查找算法来解决此问题,因为其时间复杂度优于暴力搜索的线性时间复杂度。二分查找的核心是通过比较中间元素与右端元素的大小,不断缩小搜索范围,最终找到最小值。代码示例展示了如何实现这个算法。总结中强调了二分查找适用于部分有序数组,并指出了解决这类问题的关键在于理解数组的二段单调性。

一、题干


牛客网链接


BM21 旋转数组的最小数字

https://www.nowcoder.com/practice/9f3231a991af4f55b95579b44b7a01ba




二、题解


作为查找类问题,往往通用的办法是暴力搜索。和前面的练习一样,本文不对暴力搜索算法进行过多的说明,而主要介绍二分查找算法在寻找轮转数组轮转点时的运用。暴力搜索的时间复杂度为O(N),而二分查找算法的时间复杂度为O(logN),是一种更优的解法。


二分查找:分治与不断区间划分锁定元素


一个长度为 n 的非降序数组,把一个数组最开始的若干个元素“平移”到数组的末尾,就变成了一个旋转数组。如数组 [ 1,2,3,4,5 ],经过轮转变成 [ 4,5,1,2,3 ],后两个元素成了最前面的两个元素。对于轮转数组特性更多的介绍,可以戳 👇 轮转数组问题:如何实现数组“整体逆序,内部有序”?“三步转换法”妙转数组


最基础的二分法适用于在一个已经有序数组的中寻找某一元素。本题相当于该题的变式。本题中的轮转数组被分成了有序的两个部分,如 [ 4,5,1,2,3 ] 中,虽然数组整体是无序的,但其中 [ 4,5 ] 和 [ 1,2,3 ] 仍然是有序的两个部分,在这种部分有序情况下,也可以使用二分查找法来解答,总体思路与二分法查找峰值的算法思路有些类似。


思路


1. 声明 left、right 双指针,分别指向 array 数组的最左端(首元素)与最右端(最后一个元素)。轮转数组内的最小值相当于一个“谷底”,它的左右两边的数都比它本身要大,且左侧的数一定比右侧的数还要大。我们要做的就是把这个“谷底”元素找出来。





2. 以 mid = (left + right) / 2 来表示中间值。通过比较中间值 arr[mid] 与 a[right] 的大小来不断划分搜索区间,从而锁定元素:


若 arr[mid] > arr[right] ,则说明最小元素一定在 mid 的右边部分。左界右移:left = mid + 1

若 arr[mid] < arr[right],则说明最小元素可能就在 mid 位置,或在 mid 位置的左侧。右界左移: right = mid。

若 arr[mid] == arr[right],这时就无法直接判断出最小元素的位置区间了。我们选择将右界right挨个左移,逐步缩减右界以便寻找。




代码


int minNumberInRotateArray(int* rotateArray, int rotateArrayLen ) {
 
    int left = 0;
    int right = rotateArrayLen - 1;
    int flag = 0;
    int mid = 0;
    while(left < right){
        mid = (left + right) >> 1;
        if(rotateArray[mid] > rotateArray[right]){
            left = mid + 1;
        }else if(rotateArray[mid] == rotateArray[right]){
            right--;
        }else{
            right = mid;
        }
    }
    return rotateArray[left];    
    //或return rotateArray[mid];此时while处需写while(left <= right)
}


三、总结


1. (部分)有序数组的查找问题,首先考虑使用二分法解决。其可将遍历法时间复杂度的线性级别降低至对数级别。


2. 二分查找的本质的本质是二段性,只要满足二段性的问题都可以使用二分查找求解。在本题中二分查找的二段性体现在:最小值左边的单调增,最小值右边的单调增。且左边的元素都大于右边的元素。



图源:牛客网题解



图源:牛客网题解


3. 解决分段单调问题,只需比对arr[mid],arr[0]和arr[numsSize - 1]的大小关系即可:


arr[mid] > arr[0],则最小值在 mid 位置的右半部分。

arr[mid] < arr[numsSize - 1],则最小值在 mid 位置的左半部分。

4. 必要时可以通过画图来判断边界位置的情况。


“花式”二分查找相关题型:


1.二分查找法巧得峰值


2.杨氏矩阵查找


“轮转数组”知识点:


1.轮转数组问题:如何实现数组“整体逆序,内部有序”?“三步转换法”妙转数组




相关文章
|
消息中间件 JavaScript 小程序
SpringBoot 实现 MySQL 百万级数据量导出并避免 OOM 的解决方案
SpringBoot 实现 MySQL 百万级数据量导出并避免 OOM 的解决方案
|
canal 消息中间件 关系型数据库
Canal作为一款高效、可靠的数据同步工具,凭借其基于MySQL binlog的增量同步机制,在数据同步领域展现了强大的应用价值
【9月更文挑战第1天】Canal作为一款高效、可靠的数据同步工具,凭借其基于MySQL binlog的增量同步机制,在数据同步领域展现了强大的应用价值
1549 4
|
Java
学校教师管理系统【JSP+Servlet+JavaBean】(Java课设)
学校教师管理系统【JSP+Servlet+JavaBean】(Java课设)
183 2
|
10月前
|
JSON API 开发者
构建高效API:后端开发中的RESTful最佳实践####
在数字化时代,API作为不同系统间通信的桥梁,其重要性日益凸显。本文将深入探讨RESTful API的设计原则与最佳实践,通过实际案例分析,揭示如何构建高效、可维护且易于使用的API接口,助力后端开发者提升项目质量与用户体验。 ####
|
负载均衡 监控 网络协议
在Linux中,LVS-DR模式原理是什么?
在Linux中,LVS-DR模式原理是什么?
|
供应链 Python
供需匹配(Demand-Supply Matching)的详细解释与Python代码示例
供需匹配(Demand-Supply Matching)的详细解释与Python代码示例
|
JavaScript 前端开发 Java
CocosCreator 面试题(十)Cocos Creator 内存管理
CocosCreator 面试题(十)Cocos Creator 内存管理
742 0
|
算法 数据可视化 编译器
第二代上位机开发环境搭建
欢迎来到我们的 QML & C++ 项目!这个项目结合了 QML(Qt Meta-Object Language)和 C++ 的强大功能,旨在开发出色的用户界面和高性能的后端逻辑。 在项目中,我们利用 QML 的声明式语法和可视化设计能力创建出现代化的用户界面。通过直观的编码和可重用的组件,我们能够迅速开发出丰富多样的界面效果和动画效果。同时,我们利用 QML 强大的集成能力,轻松将 C++ 的底层逻辑和数据模型集成到前端界面中。 在后端方面,我们使用 C++ 编写高性能的算法、数据处理和计算逻辑。C++ 是一种强大的编程语言,能够提供卓越的性能和可扩展性。我们的团队致力于优化代码,减少资
|
Ubuntu Linux
Pi(树莓派/香蕉派/NanoPi/香橙派)Ubuntu换KDE桌面
由于自带的Ubuntu桌面太丑了~ 所以换KDE虽然不推荐搭建使用图形,但是有时候还是需要的~ 比如当作桌面系统使用的时候 KDE #添加PPA源 sudo add-apt-repository ppa:kubuntu-ppa/backpo...
4252 0