剑指offer - 面试题11 旋转数组的最小数字

devtools/2025/2/27 4:26:15/

题目链接:旋转数组的最小数字

第一种:正确写法(num[m]和nums[r]比较)

class Solution {
public:/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可** * @param nums int整型vector * @return int整型*/int minNumberInRotateArray(vector<int>& nums) {// write code hereint l = 0, m = 0, r = nums.size() - 1;while (l < r) { // 将结果框在[l,r]的范围内,因此当l==r时,代表就是结果m = (l + r) / 2; // 此处在l=r-1时要注意死循环,因为循环条件时l < r,如果在l根据m改变时,必须给l赋值m+1,因为直接赋值为m就会导致死循环。(此处只需要注意l=m+1,而r=m是可以的,这是因为(l+r)/2的结果可能等于l,但不可能等于r)if (nums[m] > nums[r]) {l = m + 1; // 说明m是在第一个上升数组中,且m不可能是最小值,所以m这个位置不需要保留,同时为了避免死循环,l=m+1而不是l=m} else if (nums[m] < nums[r]) {r = m; // 说明m是在第二个上升数组中,且m有可能是结果的位置,因此m必须要保留,r=m而不是r=m-1} else {r--; // 此处就是第一次没想到的解法,当nums[m] == nums[r]时,没法确定是第一个还是第二个上升数组,但能确定的是,r这个位置可以不要了,因为有m在保留着结果(m不可能等于r,因为如果m==r的话,说明l==r,那么循环就走不到这里了。为了缩小结果集范围,直接r--就可以了)}}return nums[l];}
};

但是很神奇的是,上面的代码都是和num[r]比较的,但如果像下面这么写,有些用例就是失败的
第二种错误写法:nums[m]和num[l]比较

class Solution {
public:/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可** * @param nums int整型vector * @return int整型*/int minNumberInRotateArray(vector<int>& nums) {// write code hereint l = 0, m = 0, r = nums.size() - 1;while (l < r) {m = (l + r) / 2;if (nums[m] > nums[l]) {l = m + 1;} else if (nums[m] < nums[l]) {r = m;} else {l++;}}return nums[l];}
};

上面的代码表明看起来是对的,但这段代码只能通过部分用例。

因为存在特殊情况,即在旋转0个数字情况下,nums[m]是一定会大于num[l],此时按照上面的代码,l=m+1,l会越加越多,离正确答案nums[0]越来越远了。

因此这就是为什么要按照第一种写法,nums[m]和num[r]进行比较。在旋转0个数字这种特殊情况,nums[m]<num[r]是永远成立的,此时的操作正好是r = m,就不会错过正确答案。
例如[1,0,1,1,1]这个输入,nums[m]和nums[l]比较这个上面的第二种代码就是错误的。
第一轮l=0,r=4,m=2。此时nums[l]==nums[r],因此l++
第二轮l=1,r=4,m=2。此时[l,r]就是[0,1,1,1]这个升序的序列了,就是上面所说的特殊场景,nums[m] > nums[l],按照第二种代码的逻辑,l = m+1,l=3。这样直接就把正确答案跳过去了。


http://www.ppmy.cn/devtools/162961.html

相关文章

快手弹幕 websocket 分析

声明: 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 逆向分析 import timeimport requests…

Fisher信息矩阵(Fisher Information Matrix, FIM)与自然梯度下降:机器学习中的优化利器

Fisher信息矩阵与自然梯度下降&#xff1a;机器学习中的优化利器 在机器学习尤其是深度学习中&#xff0c;优化模型参数是一个核心任务。我们通常依赖梯度下降&#xff08;Gradient Descent&#xff09;来调整参数&#xff0c;但普通的梯度下降有时会显得“笨拙”&#xff0c;…

解释SSR(服务器端渲染)和CSR(客户端渲染)的区别

在现代 Web 开发中&#xff0c;SSR&#xff08;服务器端渲染&#xff09;和 CSR&#xff08;客户端渲染&#xff09;是两种主要的渲染方式。它们各自具有独特的特性、优缺点和适用场景。本文将详细探讨这两者的概念、优缺点、适用场景以及在实际开发中的应用。 1. 概念定义 1…

算法-数据结构-图的构建(邻接矩阵表示)

数据定义 //邻接矩阵表示图 //1.无向图是对称的 //2.有权的把a,到b 对应的位置换成权的值/*** 无向图* A B* A 0 1* B 1 0*/ /*** 有向图* A B* A 0 1* B 0 0*/import java.util.ArrayList; import java.util.List;/*** 带权图* A B* A 0 1* B 0 0*/ p…

muduo源码阅读:linux timefd定时器

⭐timerfd timerfd 是Linux一个定时器接口&#xff0c;它基于文件描述符工作&#xff0c;并通过该文件描述符的可读事件进行超时通知。可以方便地与select、poll和epoll等I/O多路复用机制集成&#xff0c;从而在没有处理事件时阻塞程序执行&#xff0c;实现高效的零轮询编程模…

直角三角堰计算公式

直角三角堰的计算公式通常用于确定流经直角三角形形状的堰的流量。河北瑾航科技遥测终端机 通过采集液位数据(模拟量、串口485/232)&#xff0c;计算得到瞬时流量&#xff0c;然后通过积分进行累计算出累积量&#xff1b;直角三角堰的流量计算公式为&#xff1a; 直角三角堰 计…

Java面试准备篇:全面了解面试流程与常见问题

文章目录 1.1 Java面试概述1.2 面试流程和注意事项1.3 自我介绍及项目介绍1.4 常见面试问题 在现代职场中&#xff0c;面试是求职过程中至关重要的一环&#xff0c;特别是对于Java开发者而言。为了帮助广大Java开发者更好地应对面试&#xff0c;本文将提供一份全面的Java面试…

编写第一个 C++ 程序 – Hello World 示例

“Hello World”程序是学习任何编程语言的第一步&#xff0c;也是您将学习的最直接的程序之一。它是用于演示编码过程如何工作的基本程序。您所要做的就是在输出屏幕上显示 “Hello World”。 C Hello World 程序 下面是在控制台屏幕上打印 “Hello World” 的 C 程序。 // …