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

本文介绍了如何使用二分查找算法解决旋转数组中找最小数字的问题。通过分析旋转数组的特性,利用二分查找的思想,不断缩小搜索范围,最终找到最小值。这种方法的时间复杂度为O(logN),优于暴力搜索的O(N)。文章还总结了二分查找在部分有序数组查找中的应用,并提供了相关题目的链接和代码实现。
摘要由CSDN通过智能技术生成

C语言好题方法总结。日积月累,慢慢进步!

目录

一、题干

二、题解

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

思路

 代码

三、总结


一、题干

牛客网链接

BM21 旋转数组的最小数字icon-default.png?t=M7J4https://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.轮转数组问题:如何实现数组“整体逆序,内部有序”?“三步转换法”妙转数组


评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值