Leetcode刷题笔记——二分法
二分法是搜索算法中极其典型的方法,其要求输入序列有序并可随机访问。算法思想为
输入:有序数组nums,目的数值target
要求输出:如果target存在在数组中,则输出其index,否则输出-1
- 将原数组通过[left,right]两个索引划分范围,初值left=0,right=数组的最后一个元素
- 当left <= right时
- middle = (left + right)/2
- 判断nums[middle]是不是要查找的target,如果是则返回结果
- 判断nums[middle]> target,证明要查找的target在左边,因此right = middle - 1
- 判断nums[middle]< target,证明要查找的target在右边,因此left = middle + 1
- 没有查找到return -1。
形如下图:
传统的二分法代码如下:
func binarySearch(nums []int, target int) int {
left, right := 0, len(nums)-1
for left <= right {
middle := (left + right) / 2
if nums[middle] == target {
return middle
} else if nums[middle] > target {
right = middle - 1
} else {
left = middle + 1
}
}
return -1
}
这里要注意两个问题:
- 上述算法中的第2步中
=
的判断,即for left <= right
还是for left < right
。 - 上述算法2.2-2.4中的判断条件以及下一次查找区间的设置
- 返回值代表什么意思
for left<= right 中 = 的判断
首先对于第一个问题,=
是否应该存在,取决于对于二分查找的初始化定义,例如:
- 如果二分查找遍历的区间采用
[left,right](数学中的双闭区间)
的形式,考虑left==right
即=
成立的情况,则表示区间内只有单个操作数
,这种情况还是需要处理,否则无法通过其余方式表示这种情况,所以此时=
是必须的。 - 如果二分查找遍历的区间采用
[left,right)
的形式,考虑left==right
即=
成立的情况,事实证明,这种情况并不应该存在,我们无法用[i,i)
表示任何一个区间,所以,这种情况下,=
就不是必须的。
判断条件以及下一次查找区间的设置
然后考察对于第二个问题,判断条件以及下一次查找区间应该如何设置
?
注意:二分查找是一个经典的查找算法
,其目的是查找到指定的位置或者值
,并不仅限于查找到等于target的index
这一种情况。
但无论怎样,二分查找本身有一个固定模式,即二分
,就是从middle处将区间[left,right]分成两份,然后根据middle的情况查找(或者更新新的区间),因此,我们只需要考虑清楚如下三种条件时要怎么处理即可:
- 当遍历到nums[middle] == target时应该怎样处理(新的查找区间是什么),即当前值等于目标值
- 当遍历到nums[middle] > target时应该怎样处理(新的查找区间是什么),即当前值大于目标值
- 当遍历到nums[middle] < target时应该怎样处理(新的查找区间是什么),即当前值小于目标值
讨论完上述两个问题,其实二分法就有了一个固定的框架:
func binarySearch(nums []int, target int) int {
left, right := 0, len(nums)-1
for left <= right {
middle := (left + right) / 2
if nums[middle] == target {
// 当前值等于目标值时,如何处理(新的查找区间是什么)
} else if nums[middle] > target {
// 当前值大于目标值时,如何处理(新的查找区间是什么)
} else {
// 当前值小于目标值时,如何处理(新的查找区间是什么)
}
}
// 考虑返回值的意义
return
}
返回值的含义
最后我们讨论返回值的含义
这一话题。在传统的二分查找
中,只有在两种情况下会返回:
- 查找到目标target,返回查找到的index
- 未查找到目标target,返回-1。(即文章最起始处 步骤3的含义)
这里返回值的含义表示target在nums中的index
,该值只会出现在nums[middle]==target
这一条件下。然而,刚才提到了二分查找不总是处理等式条件
,因此我们总要思考两种返回值的含义:
- nums[middle]==target,这时return代表的是什么?
- 数组中不存在target,此时return的是什么,此时left、right代表什么?
这里我们举一个稍稍复杂一点的例子对二分查找进行分析。
搜索插入位置
题目要求如下:
这个问题要求返回两种返回值:
- 在数组中找到目标值,并返回其索引
- 如果目标值不存在于数组中,返回它将会被按顺序插入的位置
其中对于情况1,传统的二分查找算法就可以解决,而情况2,则需要借助于本部分要讲解的返回值的含义
。
对于传统的二分法:
func binarySearch(nums []int, target int) int {
left, right := 0, len(nums)-1
for left <= right {
middle := (left + right) / 2
if nums[middle] == target {
return middle
} else if nums[middle] > target {
right = middle - 1
} else {
left = middle + 1
}
}
return -1
}
如果target能在nums数组中查找到,必定最终查找到一个[i,i]类型的区间,即区间中只有一个数字,否则区间就要再次进行二分。例如:如果要在下列数组中查找4所在的位置,查找过程如下,第三步时,查找区间为[2,3],有两个值,无法确定答案,则需要再次进行一次查找:
target == 4
nums 1 2 3 4
index 0 1 2 3
1 l r
2 l r
3 l r
4 lr
那么最终我们处理的情况必定是对于区间[left,right]中,其中left == right,因此middle == left == right,此时nums[middle]和target的关系。
- nums[middle] > target,则需要从middle左侧继续寻找,right = middle - 1,注意此时left = middle,left > right
- nums[middle] < target,则需要从middle右侧继续寻找,left = middle + 1,注意此时right = middle,left > right
所以此时,left指向的永远是大的那个值,right是小的那个值(因为left <= right时,循环不会终止,循环终止条件为left > right,根据数组的有序性,nums[left] > nums[right])。
最后,我们考察该题,对于数组nums,如果目标值不在其中,那么其最终查找到的值只有两种情况:
- nums[middle] < target,此时nums[middle]应该是第一个小于target的值,如果要查找target所在位置,应该返回
大于middle的index
,即left
- nums[middle] > target,此时nums[middle]应该是第一个大于target的值,如果要查找target所在位置,应该返回
等于middle的index
,用target替换middle位置的值,即left
因此,该题的结果,只需要修改传统二分查找的最后一行:
func binarySearch(nums []int, target int) int {
left, right := 0, len(nums)-1
for left <= right {
middle := (left + right) / 2
if nums[middle] == target {
return middle
} else if nums[middle] > target {
right = middle - 1
} else {
left = middle + 1
}
}
return left
}
在排序数组中查找元素的第一个和最后一个位置
题目要求如下:
注意这里查找的是元素第一次和最后一次出现的位置
,这里我们以查找第一次出现的位置举例
,后者同理。
考察我们在判断条件以及下一次查找区间的设置中强调的,考察二分查找的三种情况:
情况 | 分析 | 操作 |
---|---|---|
nums[middle] == target时,即当前值等于目标值 | 第一次出现的位置可能在当前值前面 | right = middle - 1 |
nums[middle] > target时,即当前值大于目标值 | 第一次出现的位置在当前值前面 | right = middle - 1 |
nums[middle] < target时,即当前值小于目标值 | 第一次出现的位置在当前值后面 | left = middle + 1 |
与之前不同的是当nums[middle] == target
时,不再有返回值了,那么考虑最后返回值的含义,最终left > right
时情况有如下3种:
情况 | 分析 | 操作 |
---|---|---|
nums[middle] == target | 此时,middle前的值必定<middle,而不是等于(只要等于,考虑上表的情况1,会使right = middle - 1) | return left |
nums[middle] > target | 此情况不存在,因为如果有这种情况会继续使right=middle-1 | 不进行操作 |
nums[middle] < target | 此时middle必定是target前的第一个元素 | return left |
经过上面的分析后,可以清晰的写出代码:
l, r := 0, len(nums)-1
for l <= r {
m := (l + r) / 2
if nums[m] >= target {
r = m - 1
} else {
l = m + 1
}
}
result := l
而查找元素出现的最后一个位置,只需要反过来,最后return right即可。代码如下:
l, r: = 0, len(nums)-1
for l <= r {
m := (l + r) / 2
if nums[m] <= target {
l = m + 1
} else {
r = m - 1
}
}
result := r
总结
本文详细分析了二分查找的所有细节,对于二分查找处理的问题,我们常常需要更加关注本文讨论的后两个问题:
- 判断条件以及下一次查找区间的设置
- 返回值的含义
最后填充模版即可。
func binarySearch(nums []int, target int) int {
left, right := 0, len(nums)-1
for left <= right {
middle := (left + right) / 2
if nums[middle] == target {
// 当前值等于目标值时,如何处理(新的查找区间是什么)
} else if nums[middle] > target {
// 当前值大于目标值时,如何处理(新的查找区间是什么)
} else {
// 当前值小于目标值时,如何处理(新的查找区间是什么)
}
}
// 考虑返回值的意义
return
}
Leetcode刷题笔记——二分法的更多相关文章
- LeetCode刷题笔记和想法(C++)
主要用于记录在LeetCode刷题的过程中学习到的一些思想和自己的想法,希望通过leetcode提升自己的编程素养 :p 高效leetcode刷题小诀窍(这只是目前对我自己而言的小方法,之后会根据自己 ...
- 18.9.10 LeetCode刷题笔记
本人算法还是比较菜的,因此大部分在刷基础题,高手勿喷 选择Python进行刷题,因为坑少,所以不太想用CPP: 1.买股票的最佳时期2 给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格. ...
- LeetCode刷题笔记 - 12. 整数转罗马数字
学好算法很重要,然后要学好算法,大量的练习是必不可少的,LeetCode是我经常去的一个刷题网站,上面的题目非常详细,各个标签的题目都有,可以整体练习,本公众号后续会带大家做一做上面的算法题. 官方链 ...
- LeetCode刷题笔记 - 2022
这篇博客集中整理在LeetCode的刷题记录,方便查阅 258. 各位相加 - 力扣(LeetCode) (leetcode-cn.com) 代码 class Solution { public: i ...
- Leetcode刷题笔记(双指针)
1.何为双指针 双指针主要用来遍历数组,两个指针指向不同的元素,从而协同完成任务.我们也可以类比这个概念,推广到多个数组的多个指针. 若两个指针指向同一数组,遍历方向相同且不会相交,可以称之为滑动窗口 ...
- LeetCode刷题笔记(1-9)
LeetCode1-9 本文更多是作为一个习题笔记,没有太多讲解 1.两数之和 题目请点击链接 ↑ 最先想到暴力解法,直接双循环,但是这样复杂度为n平方 public int[] twoSum(int ...
- leetcode刷题笔记
(1)Best Time to Buy and Sell Stock Total Accepted: 10430 Total Submissions: 33800My Submissions Say ...
- leetcode刷题笔记08 字符串转整数 (atoi)
题目描述 实现 atoi,将字符串转为整数. 在找到第一个非空字符之前,需要移除掉字符串中的空格字符.如果第一个非空字符是正号或负号,选取该符号,并将其与后面尽可能多的连续的数字组合起来,这部分字符即 ...
- Leetcode刷题笔记——查找
33.Search in Rotated Sorted Array 题目描述: 给定一个被翻转的整型升序数组nums,数组中无重复元素,如[4,5,6,7,0,1,2],和一个整数target.要求在 ...
- LeetCode刷题笔记(1)常用知识点
1.Integer.parseInt(String s, int radix)方法的作用是:将radix进制的字符串s转化成10进制的int型数字并返回. Integer.valueof(String ...
随机推荐
- Python竖版大屏 | 用pyecharts开发可视化的奇妙探索!
你好!我是@马哥python说,一枚10年程序猿,正在试错用pyecharts开发可视化大屏的非常规排版. 以下,我用8种ThemeType展示的同一个可视化数据大屏. 1.SHINE主题 2.LI ...
- Kubernetes 架构原则和对象设计
Kubernet¶ Kubernetes 架构原则和对象设计¶ 什么是云计算¶ 云计算平台的分类¶ 以Openstack为典型的虚拟化平台 虚拟机构建和业务代码部署分离. 可变的基础架构使后续维护风险 ...
- es笔记五之term-level的查询操作
本文首发于公众号:Hunter后端 原文链接:es笔记五之term-level的查询操作 官方文档上写的是 term-level queries,表义为基于准确值的对文档的查询,可以理解为对 keyw ...
- shader编程经典:分形--科赫曲线
序言 科赫(雪花)曲线是一个经典分形图案,来一起领略下分形之美.本篇内容用到一些基础的内容,例如UV的理解和画线技巧,有需要的话可以参考合集的画圆和画线两篇文章. 示例 shadertoy 代码: # ...
- 【论文阅读】Uformer:A General U-Shaped Transformer for Image Restoration
前言 博客主页:睡晚不猿序程 首发时间:2023.6.8 最近更新时间:2023.6.8 本文由 睡晚不猿序程 原创 作者是蒻蒟本蒟,如果文章里有任何错误或者表述不清,请 tt 我,万分感谢!orz ...
- SpringBoot进阶教程(七十六)多维度排序查询
在项目中经常能遇到,需要对某些数据集合进行多维度排序的需求.对于集合多条件排序解决方案也有很多,今天我们就介绍一种,思路大致是设置一个分值的集合,这个分值是按照需求来设定大小的,再根据分值的大小对集合 ...
- 如何在矩池云上安装和使用 Stata
Stata是一款功能强大的统计分析软件,本文提供了如何在矩池云安装使用 Stata,以及如何在 Jupyter 中使用 Stata 的简要教程. 安装 Stata 时需要确保按照官方指南进行操作,St ...
- python笔记:第六章函数&方法
1.系统函数 由系统提供,直接拿来用或是导入模块后使用 a = 1.12386 result = round(a,2) print(result) > 1.12 2.自定义函数 函数是结构化编程 ...
- AcWing 4495. 数组操作题解
思路 此题较为简单,简述一下思路. 从小到大排序,每次选取最小值,只要不为0即可 每次都为序列减去一个数字太慢,但每个数又减去的数字一样,所以可以用minus记录每个数要减去的数 C++代码 #inc ...
- Java_Day16_作业
A:简答题 1.请把我们讲解过的所有类中的方法在API中找到,并使用自己的话进行描述 答案: Map public V put(K key, V value): public void clear() ...