码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • DAY-7 | 牛客-BM21 寻找旋转数组的最小元素:二分法分治思想真的很可靠


    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挨个左移,逐步缩减右界以便寻找。

     代码

    1. int minNumberInRotateArray(int* rotateArray, int rotateArrayLen ) {
    2. int left = 0;
    3. int right = rotateArrayLen - 1;
    4. int flag = 0;
    5. int mid = 0;
    6. while(left < right){
    7. mid = (left + right) >> 1;
    8. if(rotateArray[mid] > rotateArray[right]){
    9. left = mid + 1;
    10. }else if(rotateArray[mid] == rotateArray[right]){
    11. right--;
    12. }else{
    13. right = mid;
    14. }
    15. }
    16. return rotateArray[left];
    17. //或return rotateArray[mid];此时while处需写while(left <= right)
    18. }

     


    三、总结

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


  • 相关阅读:
    轻松学习 Spring 事务
    通过Geth搭建私有以太坊网络
    mysql 之进阶查询语句
    误删记录/表/库怎么办?该如何防范误删?
    虫情测报系统——农业害虫的监测解决方案
    时空智友企业流程化管控系统 sessionid泄露漏洞 复现
    .NET开源功能强大的串口调试工具
    Mysql的事务以及存储引擎
    【Netty】1. 高性能网络通信基石-Netty入门与提高 - 笔记
    Servlet规范之预览
  • 原文地址:https://blog.csdn.net/wyd_333/article/details/126712919
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号