模块三——二分:704.二分查找

2024-04-23 05:04
文章标签 模块 二分 查找 704

本文主要是介绍模块三——二分:704.二分查找,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 二分查找算法简介
    • 特点
    • 学习中的侧重点
      • 算法原理
      • 模板
  • 题目描述
  • 算法原理
    • 解法一:暴力解法
    • 解法二:二分查找算法
      • 算法流程
      • 细节问题
        • 循环结束的条件
        • 为什么是正确的?
        • 时间复杂度
  • 代码实现

前言

本系列博客是逐渐深入的过程,建议从零开始学习的友友不要跳过一些中间的博客。

二分查找算法简介

特点

最恶心,细节最多,最容易写出死循环的算法。

学习中的侧重点

算法原理

二分查找算法一定是数组有序的情况?答案显然是否定的,只要具有二段性就能使用二分查找算法。

模板

PS:不要死记硬背,要理解性记忆。

  1. 朴素的二分模板(easy但有局限)
  2. 查找左边界的二分模板(万能,细节多)
  3. 查找右边界的二分模板(万能,细节多)
//朴素二分模板
while(left <= right){int mid = left + (right - left) / 2;//防止越界,等价于left + (right - left + 1) / 2;//PS:为偶数时前者为中间两数的左数,后者为右数if(...)left = mid + 1;else if(...)right = mid - 1;elsereturn ...;
}
//查找区间左端点的模板
while(left < right){int mid = left + (right - left) / 2;if(...)left = mid + 1;else right = mid;
}
//查找区间右端点的模板
while(left < right){int mid = left + (right - left + 1) / 2;if(...)left = mid;else right = mid - 1;

PS:分类讨论的代码,就题论题即可。

题目描述

题目链接:704.二分查找
在这里插入图片描述
题目很简单,就是在升序数组中搜索目标值target。

算法原理

解法一:暴力解法

直接遍历一遍数组即可,时间复杂度为O(N)

解法二:二分查找算法

算法流程

  • 定义 left ,right 指针,分别指向数组的左右区间。
  • 找到待查找区间的中间点 mid ,找到之后分三种情况讨论:
  1. arr[mid] == target 说明正好找到,返回 mid 的值;
  2. arr[mid] > target 说明 [mid,right] 这段区间都是⼤于 target 的,因此舍去右边区间,在左边[left, mid -1] 的区间继续查找,即让right = mid - 1 ,然后重复找mid过程;
  3. arr[mid] < target 说明 [left, mid]这段区间的值都是⼩于 target 的,因此舍去左边区间,在右边[mid + 1, right] 区间继续查找,即让 left =mid + 1 ,然后重复 找mid过程;
  • 当 left 与 right 错开时,说明整个区间都没有这个数,返回 -1 。

细节问题

循环结束的条件

left > right

为什么是正确的?

暴力解法是一次一次比较,但二分是利用了数组有序的特性,即二段性来通过比较一次来去掉多余的比较,所以暴力解法既然是正确的,那么二分也是正确的。

时间复杂度

第一次循环把区间划分成n/2,第二次是n/4,直到最坏情况下循环x次,而最后只剩下一个元素。n / 21,n / 22,n / 23…n / 2x——>2x = n——>x = logN。

代码实现

class Solution {
public:int search(vector<int>& nums, int target) {//二段性可用二分int left = 0;int right = nums.size() - 1;//结束条件while(left <= right){//直接加有可能会超出int的范围//int mid = (left + right) / 2;//换成减法防止溢出int mid = left + (right - left) / 2;if(nums[mid] > target){right = mid - 1;}else if(nums[mid] < target){left = mid + 1;}else{return mid;}}return -1;}
};

这篇关于模块三——二分:704.二分查找的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/927849

相关文章

Python中logging模块用法示例总结

《Python中logging模块用法示例总结》在Python中logging模块是一个强大的日志记录工具,它允许用户将程序运行期间产生的日志信息输出到控制台或者写入到文件中,:本文主要介绍Pyt... 目录前言一. 基本使用1. 五种日志等级2.  设置报告等级3. 自定义格式4. C语言风格的格式化方法

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图

Nginx添加内置模块过程

《Nginx添加内置模块过程》文章指导如何检查并添加Nginx的with-http_gzip_static模块:确认该模块未默认安装后,需下载同版本源码重新编译,备份替换原有二进制文件,最后重启服务验... 目录1、查看Nginx已编辑的模块2、Nginx官网查看内置模块3、停止Nginx服务4、Nginx

Python中高级文本模式匹配与查找技术指南

《Python中高级文本模式匹配与查找技术指南》文本处理是编程世界的永恒主题,而模式匹配则是文本处理的基石,本文将深度剖析PythonCookbook中的核心匹配技术,并结合实际工程案例展示其应用,希... 目录引言一、基础工具:字符串方法与序列匹配二、正则表达式:模式匹配的瑞士军刀2.1 re模块核心AP

python urllib模块使用操作方法

《pythonurllib模块使用操作方法》Python提供了多个库用于处理URL,常用的有urllib、requests和urlparse(Python3中为urllib.parse),下面是这些... 目录URL 处理库urllib 模块requests 库urlparse 和 urljoin编码和解码

创建springBoot模块没有目录结构的解决方案

《创建springBoot模块没有目录结构的解决方案》2023版IntelliJIDEA创建模块时可能出现目录结构识别错误,导致文件显示异常,解决方法为选择模块后点击确认,重新校准项目结构设置,确保源... 目录创建spChina编程ringBoot模块没有目录结构解决方案总结创建springBoot模块没有目录

idea Maven Springboot多模块项目打包时90%的问题及解决方案

《ideaMavenSpringboot多模块项目打包时90%的问题及解决方案》:本文主要介绍ideaMavenSpringboot多模块项目打包时90%的问题及解决方案,具有很好的参考价值,... 目录1. 前言2. 问题3. 解决办法4. jar 包冲突总结1. 前言之所以写这篇文章是因为在使用Mav

Python标准库datetime模块日期和时间数据类型解读

《Python标准库datetime模块日期和时间数据类型解读》文章介绍Python中datetime模块的date、time、datetime类,用于处理日期、时间及日期时间结合体,通过属性获取时间... 目录Datetime常用类日期date类型使用时间 time 类型使用日期和时间的结合体–日期时间(

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数