算法篇之二分

二分算法简介

特点

最简单的一种算法,也是最恶心,细节最多,最容易写出死循环的算法
时间复杂度O(logN)

如何学习

  1. 明白其中的算法原理,二分并不是只有数组有序的的时候使用,而是看是否具有二段性。
  2. 模板
    1. 朴素的二分模板(easy,有局限性)
    2. 查找左边界的二分模板
    3. 查找右边界的二分模板

b,c两种模板是万能模板,但是细节多

二分查找

image.png

题目链接:二分查找

算法思路:

image.png

代码
class Solution {public int search(int[] nums, int target) {int left = 0;int right = nums.length - 1;// 当left == right的时候还未判断while(left <= right) {int mid = left + (right - left) / 2;if(nums[mid] < target) {left = mid + 1;}else if(nums[mid] > target) {right = mid - 1;}else {return mid;}}return -1;}
}

朴素二分模板

image.png

在排列数组中查找元素的第一个和最后一个位置

image.png

题目链接:在排列数组中查找元素的第一个和最后一个位置

算法思路

image.png

代码
class Solution {public int[] searchRange(int[] nums, int target) {int[] result = new int[2];result[0] = result[1] = -1;if(nums.length == 0) return result;// 寻找区间左端点()int left = 0, right = nums.length - 1;// 1. left<=right会死循环,当left=right找到target时,进入循环还会走right=mid,一直循环下去。// 2. 当left==right时就是答案,不需要再判断了while(left < right) {// 如果数组是偶数要取左边元素// 因为取右边的元素会死循环,比如如果剩下(1,2)两个元素// left=1,right=2,mid=2.更新区间后,left,right,mid的值没有改变int mid = left + (right - left) / 2;// 二段性,左区间小于target,右区间大于等于targetif(nums[mid] < target) {left = mid + 1; // 由于左区间都是小于target,所以left=mid+1}else {right = mid;// 由于target在右区间间,所有right不能等于mid-1}}if(nums[left] != target) return result;result[0] = left;// 寻找区间右端点left = 0;right = nums.length - 1;while(left < right) {// 如果数组是偶数要取右边元素// 因为取左边的元素会死循环,比如如果剩下(1,2)两个元素// left=1,right=2,mid=2.更新区间后,left,right,mid的值没有改变int mid = left + (right - left + 1) / 2;if(nums[mid] <= target) {left = mid; // 由于左区间都是小于等于target,所以left不能等于mid+1}else {right = mid - 1; // 由于右区间都是小于target,所以right=mid-1}}result[1] = left;return result;}
}

模板

image.png

搜索插入位置

image.png

题目链接:搜索插入位置

算法思路

数组是有序数组,具有二段性。可以将数组划分左右两个区间,左区间<=target,右区间>target。即可以套用求右端点模板。

代码
class Solution {public int searchInsert(int[] nums, int target) {// 利用二段性,可以划分为左区间<=target,右区间>target// 也可以划分为左区间<target,右区间>=target// 咱们这里采用查找区间右端点模板int left = 0, right = nums.length - 1;while(left < right) {int mid = left + (right - left + 1) / 2;if(nums[mid] <= target) {left = mid;}else {right = mid - 1;}}if(nums[left] < target) return left + 1;return left;}
}

X的平方根

image.png

题目链接:X的平方根

算法思路

image.png

代码:
class Solution {public int mySqrt(int x) {// x的平分根一定在0~x之间long left = 0, right = x;// 变成求mid*mid=x// 因为保留整数部分,可以划分为左区间<=x,右区间>xwhile(left < right) {// 使用long,反溢出long mid = left + (right - left + 1) / 2;if(mid * mid <= x) {left = mid;}else {right = mid - 1;}}return (int)left;}
}

山脉数组的峰顶索引

image.png

题目链接:山脉数组的峰顶索引

算法思路:

image.png

代码:
class Solution {public int peakIndexInMountainArray(int[] arr) {// 利用二段性,可以将区间分为小于等于峰值的左区间,小于峰值的右区间// 即套用求区间右端点模板int left = 0, right = arr.length - 1;while(left < right) {int mid = left + (right - left + 1) / 2;if(arr[mid] > arr[mid - 1]) {left = mid;}else {right = mid - 1;}}return left;}
}

寻找峰值

image.png

题目链接:寻找峰值

算法思路:

image.png
二段性:任取一个点i,与下一个点i+1会有如下两种情况

  • nums[i] > nums[i+1]:此时「左侧区域」⼀定会存在⼭峰(因为最左侧是负⽆ 穷),那么我们可以去左侧去寻找结果;
  • nums[i] < nums[i+1]:此时「右侧区域」⼀定会存在⼭峰(因为最右侧是负⽆ 穷),那么我们可以去右侧去寻找结果;
代码:
class Solution {public int findPeakElement(int[] nums) {int left = 0, right = nums.length - 1;while(left < right) {int mid = left + (right - left) / 2;if(nums[mid] > nums[mid + 1]) {right = mid;}else {left = mid + 1;}}return left;}
}

寻找旋转排序数组中的最小值

image.png

题目链接:寻找旋转排序数组中的最小值

算法思路:

image.png
⼆分的本质:找到⼀个判断标准,使得查找区间能够⼀分为⼆。
通过图像我们可以发现, [A,B] 区间内的点都是严格⼤于 D 点的值的, C 点的值是严格小于 D 点的值的。但是当 [C,D] 区间只有⼀个元素的时候, C 点的值是可能等于 D 点的值的。
因此,初始化左右两个指针 left , right :
然后根据 mid 的落点,我们可以这样划分下⼀次查询的区间:

  • 当 mid 在 [A,B] 区间的时候,也就是 mid 位置的值严格⼤于 D 点的值,下⼀次查询区间在 [mid + 1,right] 上;
  • 当 mid 在 [C,D] 区间的时候,也就是 mid 位置的值严格⼩于等于 D 点的值,下次查询区间在 [left,mid] 上。

当区间⻓度变成 1 的时候,就是我们要找的结果。

代码:
class Solution {public int findMin(int[] nums) {int left = 0, right = nums.length - 1;int n = nums.length;while(left < right) {int mid = left + (right - left) / 2;if(nums[mid] > nums[n - 1]) {left = mid + 1;}else {right = mid;}}return nums[left];}
}

0到n-1中缺失的数字

image.png

题目链接:0到n-1中缺失的数字

算法思路:

image.png
在这个升序的数组中,我们发现:

  • 在第⼀个缺失位置的左边,数组内的元素都是与数组的下标相等的;
  • 在第⼀个缺失位置的右边,数组内的元素与数组下标是不相等的。

因此,我们可以利⽤这个「⼆段性」,来使⽤「⼆分查找」算法。

代码:
class Solution {public int takeAttendance(int[] records) {// 可以划分为左区间数值和下标相等,右区间数值和下标不相等int left = 0, right = records.length - 1;while(left < right) {int mid = left + (right - left) / 2;if(records[mid] == mid) {left = mid + 1;}else {right = mid;}}// 注意处理数组只有一个数的情况return records[left] == left ? left + 1 : left;}
}

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

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

相关文章

TCP TIME_WAIT 过多怎么处理

文章目录 1.什么是 TCP TIME_WAIT&#xff1f;2.为什么要 TIME_WAIT?3.TIME_WAIT 过多的影响4.解决办法4.1 调整短连接为长连接4.2 调整系统内核参数 5.小结参考文献 1.什么是 TCP TIME_WAIT&#xff1f; TCP 断开连接四次挥手过程中&#xff0c;主动断开连接的一方&#xff…

BeanFactory创建过程(基于Servlet)

BeanFactory创建过程&#xff08;基于Servlet&#xff09; 1. 概述1.2 那么问题就来了1.2.1 谁负责AnnotationConfigServletWebServerApplicationContext实例呢&#xff1f;1.2.2 ApplicationContextFactory初始化过程又是怎么样的&#xff1f; 1.3 总结 2. 最后 1. 概述 Anno…

什么可以用手机蓝牙控制LED???#串口通信【下】

什么可以用手机蓝牙控制LED&#xff1f;&#xff1f;&#xff1f;#串口通信【下】 前言预备知识1.小白玩串口控制的ASSII避坑1.1问题引入1.2解决问题 2.串口支持单词型指令控制2.1实现串口支持单词型指令控制的核心思路2.2利用字符数组来承接单词型指令2.3利用strstr函数来查找…

【Time Series】LSTM代码实战

一、简介 还是那句话&#xff0c;"时间序列金融"是一个很有"钱"景的话题&#xff0c;还是想尝试采用Stock时间序列预测任务DeepLearning。本文提供了LSTM预测股票的源代码。 二、算法原理 长短期记忆网络&#xff08;LSTM&#xff09;是一种特殊的循环神经…

如何衡量代码的复杂度

圈复杂度概要 最近的培训中了解到了一个概念&#xff0c;叫做圈复杂度。 圈复杂度&#xff08;Cyclomatic Complexity&#xff09;是一种衡量程序复杂度的度量方法。它由美国计算机科学家 Thomas J. McCabe 在 1976 年提出。圈复杂度通过统计程序的控制流图中的决策结构&…

linux有关安全的几个基本配置,禁止root登录,新建root权限账号

一、不安装多余的软件&#xff0c;能最小化安装就不要安装图形化界面&#xff0c;然后根据需求安装需要的软件。 二、防火墙要启用&#xff0c;如果您的这台服务器对外有服务只要放开服务就好了&#xff0c;就是说白了白名单&#xff0c;切忌一上来第一件事儿就是关闭防火墙&a…

Mobileye CES 2024 自动驾驶新技术新方向

Mobileye亮相2024年国际消费类电子产品展览会推出什么自动驾驶新技术? Mobileye再次亮相CES&#xff0c;展示了我们的最新技术&#xff0c;并推出了Mobileye DXP--我们全新的驾驶体验平台。 与往年一样&#xff0c;Mobileye是拉斯维加斯展会现场的一大亮点&#xff0c;让参观…

一文读懂Prodigal教程

2.prodigal教程 2.1 介绍 Prodigal 由橡树岭国家实验室和田纳西大学诺克斯维尔分校于2007年在能源部联合基因组研究所的主持下联合开发&#xff0c;是一种用于细菌和古细菌基因组的蛋白质编码基因预测软件工具&#xff0c;Prodigal 已成为世界上最受欢迎的微生物基因预测算法…

day28 节点操作——克隆节点、删除节点

目录 克隆节点删除节点 克隆节点 特殊情况下新增节点按以下操作&#xff1a; 复制一个原有的节点把复制的节点放入到指定的元素内部 克隆节点 元素.cloneNode(布尔值) cloneNode会克隆出一个跟原标签一样的元素&#xff0c;括号内传入布尔值 若为true&#xff0c;则代表克隆时…

Vue3批量异步更新是如何实现

文章目录 一、什么是调度执行二、如何实现可调度&#xff1f;三、批量更新 & 异步更新四、Vue原理五、最后 一、什么是调度执行 多次修改数据(例如自身num10次)&#xff0c;只进行一次页面渲染&#xff08;页面只会渲染最后一次num10&#xff09; 指的是响应式数据发生变化…

C++ copy()函数详细介绍

copy() 是一个标准库函数&#xff0c;位于 头文件中。它用于将一个容器中的元素复制到另一个容器中&#xff0c;或者将一个范围内的元素复制到另一个范围中。 函数参数介绍 copy( first, last, d_first );first 和 last&#xff1a;表示输入范围的迭代器。 first 指向要复制的…

centos7 安装 java17 安装 idea

删除旧版本的java或者说是自带的&#xff0c;免得干扰 查找java [wanglcentos7 java]$ rpm -qa|grep javajava-1.8.0-openjdk-1.8.0.262.b10-1.el7.x86_64 javapackages-tools-3.4.1-11.el7.noarch tzdata-java-2020a-1.el7.noarch python-javapackages-3.4.1-11.el7.noarch …

展台设计搭建中6个关键元素

一、哪种风格的会展展台设计更显示设计感 从已有的展台布置局面可以看出&#xff0c;不同展台设计有着不同的标准与选择原则&#xff0c;现有的一系列展台设计标识会随着现代化会展的提升重新进入更新诉求阶段。 二、展台设计一般会有那种可以选择的类别 从出现在展台设计优化阶…

React实现组件扩展机制

在java中&#xff0c;SPI机制是Java中提供的一种服务发现机制。同样&#xff0c;前端也很需要这种机制&#xff0c;这样可以做到组件可插拔&#xff0c;可替换&#xff0c;减少相互冗余。 快速使用 1.扩展点使用 通过使用Extension组件定义扩展点&#xff0c;通过name标记扩展…

2023爱分析·知识库问答市场厂商评估报告:爱数

01 研究范围定义 研究范围&#xff1a; 大模型是指通过在海量数据上依托强大算力资源进行训练后能完成大量不同下游任务的模型。2023年以来&#xff0c;ChatGPT引爆全球大模型市场。国内众多大模型先后公测&#xff0c;众多互联网领军者投身大模型事业&#xff0c;使得大模型…

C++ //练习 3.39 编写一段程序,比较两个string对象。再编写一段程序,比较两个C风格字符串的内容。

C Primer&#xff08;第5版&#xff09; 练习 3.39 练习 3.39 编写一段程序&#xff0c;比较两个string对象。再编写一段程序&#xff0c;比较两个C风格字符串的内容。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码块 /*******…

每日一道Java面试题:说一说Java中的异常

写在开头 任何一个程序都无法保证100%的正常运行&#xff0c;程序发生故障的场景&#xff0c;我们称之为&#xff1a;异常&#xff0c;在Java中对于异常的处理有一套完善的体系&#xff0c;今天我们就来一起学习一下。老样子&#xff0c;用一段简单的代码开始今天的学习。 我&a…

【CanvasKeyFrames - HTML5 Canvas 图片序列帧播放工具】

前言 一、CanvasKeyFrames 是什么&#xff1f; 用来做canvas动画的工具。 二、使用步骤 效果如图&#xff1a;上下波动的线条 1.引入库 代码如下&#xff08;示例&#xff09;&#xff1a; 在html中引入&#xff1a; <script src"canvas-keyframes.js"><…

源聚达科技:开一家抖音小店有没有风险

在数字化浪潮的推动下&#xff0c;抖音小店如雨后春笋般涌现&#xff0c;成为众多创业者眼中的香饽饽。然而&#xff0c;“盛名之下&#xff0c;其实难副”&#xff0c;开设一家抖音小店并非只有风光无限&#xff0c;其背后的风险也不容小觑。 首要的风险源自激烈的市场竞争。抖…

力扣之2629.复合函数(reduceRight )

/*** param {Function[]} functions* return {Function}*/ var compose function(functions) {return function(x) {return functions.reduceRight((result, func) > func(result), x);} };/*** const fn compose([x > x 1, x > 2 * x])* fn(4) // 9*/ 说明&#x…