大家好,我是忍者算法。今天要和大家分享一道特别有趣的题目 - LeetCode 33「搜索旋转排序数组」。这道题巧妙地将二分查找与旋转数组结合,是一道考察思维灵活性的经典题目。
想象你在看一个圆形时钟,如果把时钟的12点位置当作起点,顺时针记录1到12这些数字,这就是一个有序序列。现在,如果我们把时钟的指针从8点开始读数,到12点,再到7点,实际上就形成了一个"旋转"后的有序序列:8,9,10,11,12,1,2,3,4,5,6,7。这正是我们今天要处理的"旋转排序数组"!
题目要求:
给你一个整数数组 nums ,它原本是一个升序排序的数组,但在某个位置进行了旋转。现在给你一个目标值 target,请你在数组中搜索 target。如果存在则返回它的下标,否则返回 -1。
示例:
输入:nums = [4,5,6,7,0,1,2], target = 0
输出:4 // 0在位置4
输入:nums = [4,5,6,7,0,1,2], target = 3
输出:-1 // 3不存在于数组中
最简单的方法是直接遍历数组。虽然可以解决问题,但时间复杂度为O(n),没有利用数组的特殊性质。
既然原数组是有序的,只是被旋转了,那么旋转后的数组会有一个重要特性:它被分成了两个有序的部分。我们可以利用这个特性,结合二分查找来解决。
class Solution {
public int search(int[] nums, int target) {
if (nums == null || nums.length == 0) {
return -1;
}
int left = 0;
int right = nums.length - 1;
while (left <= right) {
int mid = left + (right - left) / 2;
if (nums[mid] == target) {
return mid;
}
// 判断哪一部分是有序的
if (nums[left] <= nums[mid]) { // 左半部分有序
if (target >= nums[left] && target < nums[mid]) {
// target在左半部分
right = mid - 1;
} else {
// target在右半部分
left = mid + 1;
}
} else { // 右半部分有序
if (target > nums[mid] && target <= nums[right]) {
// target在右半部分
left = mid + 1;
} else {
// target在左半部分
right = mid - 1;
}
}
}
return -1;
}
}
让我们来剖析这个解决方案的精妙之处:
找到有序部分
判断目标值位置
调整搜索范围
边界条件
区间判断
范围收缩
这类问题的思路可以扩展到其他场景:
寻找旋转排序数组中的最小值
旋转排序数组是否包含重复元素
在部分有序的数组中查找元素
清晰的分析过程
代码的鲁棒性
优化意识
为了帮助大家更好地理解,我来画个示意图:
<svg viewBox="0 0 800 300" xmlns="http://www.w3.org/2000/svg">
<rect width="800" height="300" fill="#f8f9fa"/>
<g transform="translate(50,50)">
<rect x="0" y="0" width="700" height="60" fill="none" stroke="#333" stroke-width="2"/>
<line x1="100" y1="0" x2="100" y2="60" stroke="#333" stroke-width="2"/>
<line x1="200" y1="0" x2="200" y2="60" stroke="#333" stroke-width="2"/>
<line x1="300" y1="0" x2="300" y2="60" stroke="#333" stroke-width="2"/>
<line x1="400" y1="0" x2="400" y2="60" stroke="#333" stroke-width="2"/>
<line x1="500" y1="0" x2="500" y2="60" stroke="#333" stroke-width="2"/>
<line x1="600" y1="0" x2="600" y2="60" stroke="#333" stroke-width="2"/>
<text x="50" y="35" text-anchor="middle" font-size="20">4text>
<text x="150" y="35" text-anchor="middle" font-size="20">5text>
<text x="250" y="35" text-anchor="middle" font-size="20">6text>
<text x="350" y="35" text-anchor="middle" font-size="20">7text>
<text x="450" y="35" text-anchor="middle" font-size="20">0text>
<text x="550" y="35" text-anchor="middle" font-size="20">1text>
<text x="650" y="35" text-anchor="middle" font-size="20">2text>
<text x="50" y="85" text-anchor="middle" font-size="16" fill="#1e88e5">lefttext>
<text x="350" y="85" text-anchor="middle" font-size="16" fill="#e91e63">midtext>
<text x="650" y="85" text-anchor="middle" font-size="16" fill="#1e88e5">righttext>
g>
<path d="M 400,140 Q 450,140 450,170" fill="none" stroke="#666" stroke-width="2" marker-end="url(#arrow)"/>
<text x="500" y="160" font-size="16">旋转点text>
<g transform="translate(50,200)">
<path d="M 0,20 H 300" stroke="#4caf50" stroke-width="3" marker-end="url(#arrow)"/>
<text x="150" y="45" text-anchor="middle" font-size="16" fill="#4caf50">左半部分有序text>
<path d="M 400,20 H 700" stroke="#ff9800" stroke-width="3" marker-end="url(#arrow)"/>
<text x="550" y="45" text-anchor="middle" font-size="16" fill="#ff9800">右半部分有序text>
g>
<defs>
<marker id="arrow" viewBox="0 0 10 10" refX="5" refY="5"
markerWidth="6" markerHeight="6" orient="auto-start-reverse">
<path d="M 0 0 L 10 5 L 0 10 z" fill="#666"/>
marker>
defs>
svg>
作者:忍者算法
公众号:忍者算法
回复【刷题清单】获取LeetCode高频面试题合集
回复【代码】获取多语言完整题解
回复【加群】加入算法交流群,一起进步
#算法面试 #LeetCode #二分查找 #数组