力扣刷题23

第一题:搜索旋转排序数组

来源: https://leetcode.cn/problems/search-in-rotated-sorted-array/description/

题目:

整数数组 nums 按升序排列,数组中的值 互不相同

在传递给函数之前,nums 在预先未知的某个下标 k0 <= k < nums.length)上进行了 向左旋转 ,使数组变为 [nums[k], nums[k+1], ..., nums[n-1], nums[0], nums[1], ..., nums[k-1]](下标 从 0 开始 计数)。例如, [0,1,2,4,5,6,7] 下标 3 上向左旋转后可能变为 [4,5,6,7,0,1,2]

给你 旋转后 的数组 nums 和一个整数 target ,如果 nums 中存在这个目标值 target ,则返回它的下标,否则返回 -1

你必须设计一个时间复杂度为 O(log n) 的算法解决此问题。

python 复制代码
class Solution:
    def search(self, nums: list[int], target: int) -> int:
        left, right = 0, len(nums) - 1
        
        while left <= right:
            mid = (left + right) // 2
            if nums[mid] == target:
                return mid
            
            # 左半部分有序
            if nums[left] <= nums[mid]:
                # target 在左半部分的有序区间内
                if nums[left] <= target < nums[mid]:
                    right = mid - 1
                else:
                    left = mid + 1
            # 右半部分有序
            else:
                # target 在右半部分的有序区间内
                if nums[mid] < target <= nums[right]:
                    left = mid + 1
                else:
                    right = mid - 1
        
        return -1

这个题让我们查找target所在位置的索引,且题目中要求时间复杂度为O(logn),所以我想到的是二分法。虽然这个数组是旋转有序(比如[4,5,6,1,2,3],但它的局部仍然是有序的 ,二分法的核心就是利用有序性缩小查找范围,这个场景刚好能适配。

二分法的本质是"分治"思路,就是每次把查找范围缩小一半,直到找到目标或者确定目标不存在。

核心条件是数组必须是有序的,可以是升序也可以是降序。

基本步骤:

初始化左指针left = 0,初始化右指针right = len(nums) - 1,覆盖整个数组;

循环:当覆盖范围有效,即left right,计算中间位置mid = (left + right) // 2,然后比较nums[mid]与target,如果相等,就是找到了目标,直接返回mid,否则分两种情况:当nums[mid] > target时,目标在左半部分,调整右指针right = mid - 1;当nums[mid] < target时,目标在右半部分,调整左指针left = mid + 1,假如循环结束后仍然没有找到答案,就返回-1。


举个例子模拟一下,以nums = [4,5,6,7,0,1,2],target = 0为例:

初始left = 0,right = 6,mid = 3,nums[mid] = 7 ,不等于target,nums[mid] > target, 0 < 7,且左半部分有序,但0不在此部分内,所以left = 4;

此时,left = 4,right = 6,mid = 5,nums[mid] = 1,nums[mid] > target,0 < 1,且 右半部分有序,但0也不在此部分内,所以right = 4;

此时left = 4,right = 4,mid = 4,nums[mid] = 0,nums[mid] == target,返回mid = 4。

与示例所给结果一致。

ok,记录完毕。


第二题:在排序数组中查找元素的第一个和最后一个位置

来源: https://leetcode.cn/problems/find-first-and-last-position-of-element-in-sorted-array/

题目:

给你一个按照非递减顺序排列的整数数组 nums,和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。

如果数组中不存在目标值 target,返回 [-1, -1]

你必须设计并实现时间复杂度为 O(log n) 的算法解决此问题。

python 复制代码
from typing import List

class Solution:
    def searchRange(self, nums: List[int], target: int) -> List[int]:
        def find_left():
            left, right = 0, len(nums) - 1
            while left <= right:
                mid = (left + right) // 2
                if nums[mid] < target:
                    left = mid + 1
                else:
                    right = mid - 1
            if left < len(nums) and nums[left] == target:
                return left
            else:
                return -1

        def find_right():
            left, right = 0, len(nums) - 1
            while left <= right:
                mid = (left + right) // 2
                if nums[mid] > target:
                    right = mid - 1
                else:
                    left = mid + 1
            if right >= 0 and nums[right] == target:
                return right
            else:
                return -1

        left_pos = find_left()
        if left_pos == -1:
            return [-1, -1]
        right_pos = find_right()
        return [left_pos, right_pos]

根据题目提示,时间复杂度为O(logn),所以我想到的还是二分法。原理和上一题一样,搞定,记录完毕。

相关推荐
菜鸡儿齐1 小时前
leetcode-子集
算法·leetcode·深度优先
今儿敲了吗1 小时前
28| A-B数对
数据结构·c++·笔记·学习·算法
Desirediscipline1 小时前
#include<limits>#include <string>#include <sstream>#include <iomanip>
java·开发语言·前端·javascript·算法
Felven2 小时前
B. Luntik and Subsequences
算法
菜鸡儿齐2 小时前
leetcode-括号生成
算法·leetcode·职场和发展
fs哆哆2 小时前
在VB.NET中,随机打乱列表顺序的算法与方法
算法·.net
pen-ai2 小时前
【Yolo系列】Yolov3 目标检测算法原理详解
算法·yolo·目标检测
田里的水稻2 小时前
EP_基于UWB和单线激光雷达的托盘转送
人工智能·算法·数学建模·机器人·自动驾驶
List<String> error_P2 小时前
DFS(深度优先搜索)
数据结构·算法·dfs