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.轮转数组问题:如何实现数组“整体逆序,内部有序”?“三步转换法”妙转数组




相关文章
|
8月前
【每日一题Day149】LC2389和有限的最长子序列 | 贪心+前缀和+二分查找
【每日一题Day149】LC2389和有限的最长子序列 | 贪心+前缀和+二分查找
50 0
【LeetCode-每日一题】-378. 有序矩阵中第K小的元素
【LeetCode-每日一题】-378. 有序矩阵中第K小的元素
|
5月前
|
算法 索引
【算法】二分算法——山脉数组的峰顶索引
【算法】二分算法——山脉数组的峰顶索引
|
7月前
每日一题 540. 有序数组中的单一元素
每日一题 540. 有序数组中的单一元素
|
7月前
|
存储 算法
算法训练,牛客.判断是不是平衡二叉树 牛客.最大子矩阵两个数组的交集牛客.数组中两个字符串的最小距离
算法训练,牛客.判断是不是平衡二叉树 牛客.最大子矩阵两个数组的交集牛客.数组中两个字符串的最小距离
算法训练,牛客.判断是不是平衡二叉树 牛客.最大子矩阵两个数组的交集牛客.数组中两个字符串的最小距离
|
8月前
【编程题-错题集】连续子数组最大和(动态规划 - 线性 dp)
【编程题-错题集】连续子数组最大和(动态规划 - 线性 dp)
|
8月前
|
搜索推荐 算法 索引
【排序算法】深入解析快速排序(霍尔法&&三指针法&&挖坑法&&优化随机选key&&中位数法&&小区间法&&非递归版本)
【排序算法】深入解析快速排序(霍尔法&&三指针法&&挖坑法&&优化随机选key&&中位数法&&小区间法&&非递归版本)
207 4
|
8月前
|
人工智能 算法
前缀和算法题(区间次方和、小蓝平衡和、大石头的搬运工、最大数组和)
前缀和算法题(区间次方和、小蓝平衡和、大石头的搬运工、最大数组和)
|
8月前
|
算法 测试技术 C#
【动态规划】【数论】【区间合并】3041. 修改数组后最大化数组中的连续元素数目
【动态规划】【数论】【区间合并】3041. 修改数组后最大化数组中的连续元素数目
|
8月前
|
测试技术
【动态规划】【记忆化搜索】【状态压缩】1681. 最小不兼容性
【动态规划】【记忆化搜索】【状态压缩】1681. 最小不兼容性