二分:二分查找、在排序数组中查找元素的第一个和最后一个位置、搜索插入位置、x 的平方根 - 实践

news/2026/1/18 22:35:28/文章来源:https://www.cnblogs.com/gccbuaa/p/19499363

  • 二分查找
    • 题目描述
    • 算法原理
    • 算法实现
  • 在排序数组中查找元素的第一个和最后一个位置
    • 题目描述
    • 算法原理
    • 算法实现
  • 搜索插入位置
    • 题目描述
    • 算法原理
    • 算法实现
  • x 的平方根
    • 题目描述
    • 算法原理
    • 算法实现

二分查找

题目描述

二分查找
给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果 target 存在返回下标,否则返回 -1。

你必须编写一个具有== O(log n)== 时间复杂度的算法。

示例 1:

输入: nums = [-1,0,3,5,9,12], target = 9
输出: 4
解释: 9 出现在 nums 中并且下标为 4

示例 2:

输入: nums = [-1,0,3,5,9,12], target = 2
输出: -1
解释: 2 不存在 nums 中因此返回 -1

提示:

算法原理

  • 算法一:枚举法

扫描遍历一遍,时间复杂度自然为O(n).

  • 算法二:二分查找:

二分查找作为一款经久不衰的算法自然有其魅力所在。
我们以输入: nums = [-1,0,3,5,9,12], target = 9为例

在这里插入图片描述
可以看到mid指针指向的数小于target,根据单调性。mid本身和mid左边的数都是小于target的,因此可以排除:
在这里插入图片描述
于是下次查找就出答案了。
可以看到我们二分查找一次干掉一半搜索区间,因此时间复杂度为O(logn).

实际上我们可以证明二分的时候,平均操作次数是最低的:
我们不妨设target等可能出现在每个数上,即每个数是target的可能为1/n.
我们使用k分查找,即将数组分成k叉树。树高为[logkn].
第i层的结点数量为ki.
因此查找次数期望为
1n∑i=0logkn(i+1)ki\frac{1}{n}\sum_{i=0}^{log_kn}(i+1)k^in1i=0logkn(i+1)ki
然后经过化简可以得到,k取2的时候得最小值。

算法实现

class Solution {
public:
int search(vector<int>& nums, int target){int left = 0, right = nums.size() - 1, mid = 0;while (left <= right){mid = left + (right - left) / 2;if (nums[mid] == target)return mid;else if (nums[mid] < target)left = mid + 1;elseright = mid - 1;}return -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]

提示:

算法原理

  • 二分查找

既然要求O(logn)的时间复杂度那么就不能用枚举法了。我们直接用二分查找来做。
我们可以将排序数组看成三个区间:
在这里插入图片描述
那么我们现在相当于求绿色区间的左端点和右端点,这需要分开讨论。
先来看左端点:
在这里插入图片描述
可以看到我们左端点可以将数组划分成两个区间,左边是<target,右边是>=target。其中我们的begin是右边区间的元素。

因此当我们的mid指向左区间元素时,mid势必不是最终结果。因此令left=mid+1.
但是当mid指向右区间元素时,mid可能是最终结果,因此需要保留令right=mid

这时候也意味着我们取中点mid=left+(right-left)/2,必须向下取整
这是个相当显然的事实。例如区间为[1,3]target=3.如果采取向上取整的算法求中点,我们可以算得mid=1.于是right和left不变。继续求的mid=1.就进入了死循环。

接下来看求右端点:
在这里插入图片描述

同样的我们的右端点把区间分成<=target和>target.
当mid落在左区间时有可能是最终结果,因此left=mid。
当mid落在右区间时,势必不可能为最终结果,因此right=mid-1.
这时候我们自然要采取向上取整的方式求中点,理由同上。

  • 简而言之,当左区间有可能是最终结果时,我们要确保取中点时有向右区间的趋势,因此向上取整;当右区间有可能是最终结果时,我们要确保取中点时有向左区间的趋势,因此向下取整。

算法实现

class Solution {
public:
vector<int> searchRange(vector<int>& nums, int target){//处理边界if (!nums.size())return { -1,-1 };int left = 0, right = nums.size() - 1, begin = 0, mid = 0;//找左边界while (left < right){mid = left + (right - left) / 2;if (nums[mid] < target)left = mid + 1;elseright = mid;}if (nums[right] != target)return { -1,-1 };begin = left;left = 0;right = nums.size() - 1;//找右边界while(left<right){mid = left + (right - left + 1) / 2;if (nums[mid] > target)right = mid - 1;elseleft = mid;}return { begin,right };}};

在这里插入图片描述

搜索插入位置

题目描述

搜索插入位置
给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

请必须使用时间复杂度为 O(log n) 的算法。

示例 1:

输入: nums = [1,3,5,6], target = 5
输出: 2

示例 2:

输入: nums = [1,3,5,6], target = 2
输出: 1

示例 3:

输入: nums = [1,3,5,6], target = 7
输出: 4

提示:

算法原理

  • 二分查找

根据题目,我们知道我们要找的是第一个>=target的元素的位置。
即将数组化成成<target和>=target的两个区间,而我们要求右区间的左端点。参考上一题的求左端点算法即可.

算法实现

class Solution {
public:
int searchInsert(vector<int>& nums, int target){if (target > nums.back())return nums.size();int left = 0, right = nums.size() - 1, mid = 0;while (left < right){mid = left + (right - left) / 2;if (nums[mid] >= target)right = mid;elseleft = mid + 1;}return right;}};

在这里插入图片描述

x 的平方根

题目描述

x 的平方根
给你一个非负整数 x ,计算并返回 x 的 算术平方根 。

由于返回类型是整数,结果只保留 整数部分 ,小数部分将被 舍去 。

注意:不允许使用任何内置指数函数和算符,例如 pow(x, 0.5) 或者 x ** 0.5 。

示例 1:

输入:x = 4
输出:2

示例 2:

输入:x = 8
输出:2
解释:8 的算术平方根是 2.82842…, 由于返回类型是整数,小数部分将被舍去。

提示:

算法原理

  • 算法一:枚举法

自1开始枚举各种可能,时间复杂度为O(n)

  • 算法二:二分查找

输入x,相当于我们在[1,x]区间中找到一个最大的数pow使得pow*pow<=x.
即我们可以将[1,x]划分成两个区间:pow*pow<=x和pow*pow>x。
而我们要找的就是左区间的右端点。参考上上一题算法即可。

算法实现

class Solution {
public:
int mySqrt(int x)
{
int left = 0, right = x, mid = 0;
while (left < right)
{
mid = left + (right - left) / 2 + (right - left) % 2;
if (mid  <= x/mid)
left = mid;
else
right = mid - 1;
}
return left;
}
};

在这里插入图片描述

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

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

相关文章

极简风装修不踩坑!3家宝藏装修公司实测推荐,高级感拉满 - 品牌测评鉴赏家

极简风装修不踩坑!3家宝藏装修公司实测推荐,高级感拉满一、开篇:为什么极简风超火,却 90% 的人装成 “廉价毛坯房”? (一)极简风的治愈魔力:当代人的心灵避风港 在这个快节奏、高压力的时代,我们每天穿梭于城…

苏州二手房局部改造不踩坑!6家宝藏装修公司,老房焕新超省心 - 品牌测评鉴赏家

苏州二手房局部改造不踩坑!6家宝藏装修公司,老房焕新超省心一、开篇:苏州老房焕新,局部改造才是性价比之王 (一)为什么苏州二手房更适合局部改造? 走在苏州的大街小巷,随处可见充满历史韵味的老房子。这些二手…

2026毛坯房装修不迷路!这些品牌闭眼选 - 品牌测评鉴赏家

2026毛坯房装修不迷路!这些品牌闭眼选装修前的灵魂拷问 当你满心欢喜拿到毛坯房钥匙,准备大干一场时,是不是瞬间被各种难题砸晕?从何下手?选什么装修风格?怎么把控预算?找施工队还是装修公司?这些问题像紧箍咒…

【工具分享】--编写POC之Wavely的使用

最近面试有时候会被问到是否编写过poc&#xff0c;由于我真正手写的经验其实有点不足所以第一次被问到时候是有点懵逼的&#xff0c;所以有了这篇poc总结&#xff08;当然啦&#xff0c;这是我回过头又复习了许多篇有关poc的文章并且结合了我自己常用的工具再输出的文章&#x…

【毕业设计】基于python-CNN-pytorch深度学习训练识别T恤的颜色

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

day146—递归—验证二叉搜索树(LeetCode-98)

题目描述给你一个二叉树的根节点 root &#xff0c;判断其是否是一个有效的二叉搜索树。有效 二叉搜索树定义如下&#xff1a;节点的左子树只包含 严格小于 当前节点的数。节点的右子树只包含 严格大于 当前节点的数。所有左子树和右子树自身必须也是二叉搜索树。示例 1&#x…

【ST表】洛谷 P3865 【模板】ST 表 RMQ 问题

View Post【ST表】洛谷 P3865 【模板】ST 表 & RMQ 问题题目 https://www.luogu.com.cn/problem/P3865 题解 ST表(Sparse Table,稀疏表)主要用来解决 RMQ(区间最大/最小值查询)问题。主要应用倍增思想,可以实…

2026苏州100平左右新房装修指南:高性价比公司全揭秘 - 品牌测评鉴赏家

2026苏州100平左右新房装修指南:高性价比公司全揭秘一、100 平新房装修痛点大剖析 在苏州,100 平左右的新房大多是紧凑三居或舒适两居 ,看似面积刚刚好,可一旦进入装修环节,各种麻烦事就接踵而至。很多朋友都向我…

2026苏州二手房局部翻新大揭秘!这些公司你不能错过 - 品牌测评鉴赏家

2026苏州二手房局部翻新大揭秘!这些公司你不能错过一、苏州二手房翻新热潮 在当下的苏州房地产市场中,二手房的交易热度持续攀升。数据显示,苏州二手房市场成交量在近期呈现出明显的增长态势,越来越多的人选择购入…

苏州装修公司口碑大揭秘!这几家名列前茅 - 品牌测评鉴赏家

苏州装修公司口碑大揭秘!这几家名列前茅一、装修公司的重要性 家,是我们心灵的避风港,是生活中最温暖的存在。而装修,则是赋予这个避风港独特魅力和舒适体验的关键环节。选择一家靠谱的装修公司,对于打造理想家园…

Go 语言 GMP 调度模型深度解析 - 教程

pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", …

HTML一键打包EXE工具2.2.0版本重磅更新 - 2026年最新版本稳定性大幅提升

HTML打包EXE工具迎来2026年首个重要版本更新&#xff01;2.2.0版本专注于稳定性提升和用户体验优化&#xff0c;修复了多个影响使用的关键问题&#xff0c;新增清理本地激活数据功能&#xff0c;为开发者提供更可靠的HTML转EXE解决方案。 CSDN免积分下载最新版本 【免费】HTM…

2024年9月GESP真题及题解(C++七级): 小杨寻宝

2024年9月GESP真题及题解(C七级): 小杨寻宝 题目描述 小杨有一棵包含 nnn 个节点的树&#xff0c;树上的一些节点放置有宝物。 小杨可以任意选择一个节点作为起点并在树上移动&#xff0c;但是小杨只能经过每条边至多一次&#xff0c;当小杨经过一条边后&#xff0c;这条边就…

苏州装修性价比大揭秘!哪家公司才是真王者? - 品牌测评鉴赏家

苏州装修性价比大揭秘!哪家公司才是真王者?一.苏州装修市场现状 在苏州,装修市场可谓是一片繁荣景象,各类装修公司如雨后春笋般涌现,数量众多让人眼花缭乱。据不完全统计,苏州大大小小的装修公司不下数百家 ,从…

HBase与Flink CDC:实时数据同步技术

HBase与Flink CDC:实时数据同步技术 关键词:HBase、Flink CDC、实时数据同步、变更数据捕获、分布式系统、数据集成、增量处理 摘要:本文深入探讨基于HBase与Flink CDC的实时数据同步技术体系。首先解析HBase存储架构与Flink CDC核心原理,通过数学模型论证数据一致性保障机…

大数据环境下空间数据分析的最佳实践

大数据时代空间数据分析&#xff1a;从踩坑到落地的7个最佳实践 引言&#xff1a;你是否被“海量空间数据”卡住了&#xff1f; 做外卖平台的朋友跟我吐槽&#xff1a;“我们有100万骑手的轨迹数据&#xff0c;想分析他们的停留热点&#xff0c;用ArcGIS跑了3天还没出结果&…

学长亲荐10个AI论文网站,继续教育学生轻松搞定论文格式!

学长亲荐10个AI论文网站&#xff0c;继续教育学生轻松搞定论文格式&#xff01; AI 工具如何让论文写作更轻松 在当今信息爆炸的时代&#xff0c;继续教育学生面对论文写作时常常感到压力山大。无论是格式要求、内容逻辑还是语言表达&#xff0c;都可能成为阻碍完成论文的“拦路…

2026年诚信的西山区心理咨询,昆明心理咨询,南市区心理咨询公司行业优质名录 - 品牌鉴赏师

引言在当今社会,心理健康问题日益受到人们的关注,心理咨询行业也随之蓬勃发展。在昆明西山区、南市区等地,心理咨询公司如雨后春笋般涌现。为了帮助消费者在众多的心理咨询公司中挑选出优质、诚信的机构,我们依据国…

微信小程序毕设项目推荐-基于springboot的保护濒危动物公益网站系统公益网站建设、动物保护系统、濒危物种网站【附源码+文档,调试定制服务】

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

2024年9月GESP真题及题解(C++七级): 矩阵移动

2024年9月GESP真题及题解(C七级): 矩阵移动 题目描述 小杨有一个 nmn \times mnm 的矩阵&#xff0c;仅包含 01? 三种字符。矩阵的行从上到下编号依次为 1,2,…,n1,2,\dots, n1,2,…,n&#xff0c;列从左到右编号依次为 1,2,…,m1, 2, \dots, m1,2,…,m。小杨开始在矩阵的左上…