力扣刷题--数组--第二天

news2024/11/28 1:39:20

  今天仍然做二分查找相关的题目。先来回顾一下二分查找的方法和使用的条件。二分查找是在数组中查找目标值的一种方法,通过边界索引确定中间索引,判断中间索引处的元素值和目标值的大小,来不断缩小查找区间。使用二分查找有如下一些限制:

  • 数组是单调递增或单调递减;
  • target值唯一。

话不多说,直接上题!

1.题目链接

题目名称:在排序数组中查找元素的第一个和最后一个位置
题目详情:
  非减顺序排列的整数数组 nums,和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。
  如果数组中不存在目标值 target,返回 [-1, -1]。
  你必须设计并实现时间复杂度为 O(log n) 的算法解决此问题。

示例 1:
输入:nums = [5,7,7,8,8,10], target = 8
输出:[3,4]
示例 2:

输入:nums = [5,7,7,8,8,10], target = 6
输出:[-1,-1]

示例 3:
输入:nums = [], target = 0
输出:[-1,-1]

解法一:我自己写出来的!明天奖励自己吃个鸡腿,哈哈哈哈
主要思路:
  首先题目中给定的数组是非减顺序,其实不符合二分查找的条件,但是题目最终是要找到等于target的开始位置和结束位置。这里可以分为三种情况:
  (1) target不在数组中,则和普通的二分查找方式一样;
  (2) target在数组中,但有且仅有一个,那也和普通的二分查找一样;
  (3) target在数组中,但个数不止一个,这种情况下当nums[mid] == target时,首先就可以把mid存储起来,其次因为在当前这个区间内不止一个target,所以不能按照以前的方式直接返回或确定当前mid左侧的值一定全部小于target,mid右侧的值一定全部大于target。故这里就会以mid为节点,分成两个子区间[lindex,mid-1]和[mid+1,rindex],两个子区间分别再进行二分查找,寻找满足条件的索引。

class Solution:
    def searchRange(self, nums: List[int], target: int) -> List[int]:
    	# 设定一个list,用来存储数组中等于target的索引
    	# 且out要么为空要么有且仅有两个元素,包括左边界和右边界
        out=[]  
        def run(lindex,rindex):
            if lindex > rindex:
                return -1
            mid=lindex+(rindex-lindex)//2
            # 当mid索引元素不等于target时,像之前普通二分查找法一样
            if nums[mid] != target:
                if nums[mid]>target:
                    rindex=mid-1
                else:
                    lindex=mid+1
                return run(lindex,rindex)
            # 当mid索引元素==target时
            else:
                if len(out) == 0:
                	# 若当前mid是第一个判断为target的索引,则填充至out
                    out.extend([mid,mid]) 
                else:
                	# 当满足条件的索引大于1个时,
                	# 则这里判断当前索引是左边界还是右边界
                    out[0]=min(out[0],mid)
                    out[1]=max(out[1],mid)
                # 当mid索引==target时,需将[lindex,rindex]划分成两个子区间
                # 分别进行二分查找,继续搜索==target的索引
                run(lindex,mid-1)
                run(mid+1,rindex)

        run(0,len(nums)-1)
        if len(out) == 0:
            return [-1,-1]
        return out

在这里插入图片描述

解法二:看的代码随想录的解法,可以学习这种思路
解题思路:
  这里使用两次二分法分别去查找target的左边界和右边界(特别注意的是,这里的左边界指的是第一个target索引的前面一位索引,右边界也是一样),也可以分为三种情况去讨论,分别是:
  (1) target小于数组的最小值或大于数组的最大值,即在数组范围的左侧或右侧;
  (2) target在数组范围内,但target不在数组中;
  (3) target在数组中。

class Solution:
    def searchRange(self, nums: List[int], target: int) -> List[int]:
        # 获得右边界
        def  getRightBorder(nums,target):
            rightBorder=-2
            lindex=0
            rindex=len(nums)-1
            while lindex<=rindex:
                mid=lindex+(rindex-lindex)//2
                if nums[mid] > target:
                    rindex=mid-1
                else:
                    lindex=mid+1
                    rightBorder=lindex  # 这里右边界更新
            return rightBorder
        # 获得左边界
        def getLeftBorder(nums,target):
            leftBorder=-2
            lindex=0
            rindex=len(nums)-1
            while lindex<=rindex:
                mid=lindex+(rindex-lindex)//2
                if nums[mid] >= target:
                    rindex=mid-1
                    leftBorder=rindex  # 这里左边界更新
                else:
                    lindex=mid+1  
            return leftBorder            

        leftBorder = getLeftBorder(nums,target)
        rightBorder = getRightBorder(nums,target)
        # 情况一,target不在数组范围内时,以target在数组左侧为例,
        # 二分查找会不断更新rindex的值,那么leftBorder不更新,一直为-2。同理可以推出,target在数组右侧的情况。
        if leftBorder == -2 or rightBorder == -2:
            return [-1,-1]
        # 情况三,因为leftBorder和rightBorder分别位于target的前一个索引位置和后一个索引位置
        # 故 rightBorder-leftBorder必定大于1
        elif rightBorder-leftBorder > 1:
            return [leftBorder+1,rightBorder-1]
        # 情况二
        else:
            return [-1,-1]

参考

代码随想录-34. 在排序数组中查找元素的第一个和最后一个位置

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1650353.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PXE批量安装

系统装机的三种引导方式 u盘光盘网络装机 光盘&#xff1a; 1.类似于usb模式 2.刻录模式 系统安装过程 加载boot loader Boot Loader 是在操作系统内核运行之前运行的一段小程序。通过这段小程序&#xff0c;我们可以初始化硬件设备、建立内存空间的映射图&#xff0c;从…

使用开放式用户通信连接两台西门子S71200plc

步骤1.在项目中创建两台PLC。 步骤2.分别设置两个PLC的参数。 plc1 plc2 步骤3.对两个plc进行组态 步骤4.在plc1和plc2中各自创建DB块&#xff0c;用于通信。 须在块的属性中取消优化块的访问选项。 plc1 plc2 步骤5.往plc1的main块中编写代码。 步骤6.往plc2的main块中编写…

AndroidStudio的Iguana版的使用

1.AndroidStudio介绍 Android Studio 是用于开发 Android 应用的官方集成开发环境 (IDE)。Android Studio 基于 IntelliJ IDEA 强大的代码编辑器和开发者工具&#xff0c;还提供更多可提高 Android 应用构建效率的功能&#xff0c;例如&#xff1a; 基于 Gradle 的灵活构建系统…

XORM 框架的使用

1、xorm 1.1、xorm 简介 xorm 是一个简单而强大的Go语言ORM库. 通过它可以使数据库操作非常简便。 特性 支持 struct 和数据库表之间的灵活映射&#xff0c;并支持自动同步事务支持同时支持原始SQL语句和ORM操作的混合执行使用连写来简化调用支持使用ID, In, Where, Limit,…

nginx模型设计和进程讲解

一. Nginx进程模型解析 1. master主进程 和 worker工作进程 [rootlocalhost sbin]# ps -ef|grep nginx root 15411 1 0 21:08 ? 00:00:00 nginx: master process ./nginx nobody 15412 15411 0 21:08 ? 00:00:00 nginx: worker process root…

pytest教程-39-钩子函数-pytest_runtest_setup

领取资料&#xff0c;咨询答疑&#xff0c;请➕wei: June__Go 上一小节我们学习了pytest_runtest_protocol钩子函数的使用方法&#xff0c;本小节我们讲解一下pytest_runtest_setup钩子函数的使用方法。 pytest_runtest_setup 钩子函数在每个测试用例的 setup 阶段被调用。这…

学习软考----数据库系统工程师22

关系运算 基本的关系代数运算 拓展的关系运算 除&#xff1a;需要S连接中属性为C和D的两个元组都与R连接一样&#xff0c;且在R连接中对应的另外的元素完全一致 总结

聊聊 ASP.NET Core 中间件(三):如何创建自己的中间件?

前言 本质上&#xff0c;中间件类也是一个普通的 .NET 类&#xff0c;它不需要继承任何父类或者实现任何接口。 但是有几个约定&#xff1a; 需要有一个构造方法构造方法至少要有一个 RequestDelegate 类型的参数&#xff0c;用来指向下一个中间件。需要定义一个名字为 Invo…

交易复盘-20240507

仅用于记录当天的市场情况&#xff0c;用于统计交易策略的适用情况&#xff0c;以便程序回测 短线核心&#xff1a;不参与任何级别的调整&#xff0c;采用龙空龙模式 一支股票 10%的时候可以操作&#xff0c; 90%的时间适合空仓等待 蔚蓝生物 (5)|[9:25]|[36187万]|4.86 百合花…

STM32F10x移植FreeRTOS

一、获取FreeRTOS源码 &#xff08;1&#xff09;登录FreeRTOS官网&#xff1a;www.freertos.org&#xff0c;下载第一个压缩包 &#xff08;2&#xff09;通过GitHub网站&#xff1a;github.com/FreeRTOS/FreeRTOS下载&#xff0c;由于该网站服务器在国外&#xff0c;所以访问…

28.leetcode---前K个高频单词(Java版)

题目链接: https://leetcode.cn/problems/top-k-frequent-words/description/ 题解: 代码: 测试:

巨控GRM561/562/563/564Q杀菌信息远程监控

摘要 通过程序编写、手机APP画面制作等运行系统&#xff0c;实现电脑及手机APP显示的历史曲线画面和数据图形化的实时性。 不仅流程效率提升90%以上&#xff0c;同时为杀菌生产提供有利的质量保障&#xff0c;还有效规避因触屏及内存卡的突发异常导致历史数据的丢失&#xff0…

强大的禄得可转债自定义因子轮动系统完成,可转债三低为例子

经过几天的测试终于完成了可转债自定义因子轮动&#xff0c;超过1000行的源代码 我提供了服务器的数据支持自动api下载&#xff0c;我给大家维护数据 网页 http://120.78.132.143:8023/ 录得数据支持http://120.78.132.143:8023/lude_data_app api数据支持&#xff0c;我提供…

杭州恒生面试,社招,3年经验

你好&#xff0c;我是田哥 一位朋友节前去恒生面试&#xff0c;其实面试问题大部分都是八股文&#xff0c;但由于自己平时工作比较忙&#xff0c;完全没有时间没有精力去看八股文&#xff0c;导致面试结果不太理想&#xff0c;HR说节后通知面试结果&#xff08;估计是凉了&…

传统汽车空调系统工作原理

1.首先讲一个概念 液体变成气体&#xff1a;吸热 气体变成液体&#xff1a;放热 2.在汽车空调系统中热量的传递的介质不是水&#xff0c;而是氟利昂&#xff0c;简称&#xff1a;“氟”。 3.传统式汽车空调结构如下 该三个部件位于车头进气口位置 该部位位于汽车驾驶车厢前方…

常见的容器技术有哪些

容器技术是一种轻量级的软件封装方式&#xff0c;它将软件代码及其依赖项打包在一起&#xff0c;这样应用可以在任何支持容器的系统上无缝运行。它允许应用程序及其依赖项在一个隔离的环境中运行&#xff0c;这个环境被称为容器。容器技术有助于提高应用程序的可移植性、一致性…

Vue3专栏项目 -- 一、第一个页面(上)

一、ColumnList 组件&#xff08;专栏列表组件&#xff09;编码&#xff1a; 该组件要接收一个数组&#xff0c;数组中是一个个专栏数据&#xff0c;数据中包括id、title、avator、description。所以我们定义一个泛型&#xff0c;泛型为id为number类型title为string类型如下这…

初识C++ · 类和对象(下)

目录 1 再谈构造函数 2 类中的隐式类型转换 3 Static成员 4 友元和内部类 5 匿名对象 6 编译器的一些优化 1 再谈构造函数 先看一段代码&#xff1a; class Date { public :Date(int year, int month, int day){_year year;_month month;_day day;} private:int _ye…

Redis之Linux下的安装配置

Redis之Linux下的安装配置 Redis下载 Linux下下载源码安装配置 方式一 官网下载&#xff1a;https://redis.io/download ​ 其他版本下载&#xff1a;https://download.redis.io/releases/ 方式二&#xff08;推荐&#xff09; GitHub下载&#xff1a;https://github.com/r…

基于ambari hdp的kafka用户授权读写权限

基于ambari hdp的kafka用户授权读写权限 版本Kafka 2.0.0添加自定义配置修改admin密码重启kafka授权读取授权写入有效通配符部分举例 版本Kafka 2.0.0 添加自定义配置 authorizer.class.name kafka.security.auth.SimpleAclAuthorizer super.users User:admin allow.everyo…