算法每日双题精讲 —— 二分查找(山脉数组的峰顶索引,寻找峰值)

news2025/1/28 1:11:49

 🌟快来参与讨论💬,点赞👍、收藏⭐、分享📤,共创活力社区。 🌟 

别再犹豫了!快来订阅我们的算法每日双题精讲专栏,一起踏上算法学习的精彩之旅吧💪   


        在算法的广袤世界里,二分查找算法凭借其高效性与独特的解题思路,成为众多开发者和算法爱好者的得力工具。今天,让我们一同深入研究 “山脉数组的峰顶索引” 以及 “寻找峰值” 这两道经典题目,探索二分查找算法在其中的巧妙应用。

目录

一、山脉数组的峰顶索引

📖题目描述

🧠讲解算法原理

💻代码实现(以C++为例)

复杂度分析

二、寻找峰值

📖题目描述

🧠讲解算法原理

💻代码实现(以 C++ 为例)

复杂度分析


一、山脉数组的峰顶索引

题目链接👉【力扣】

📖题目描述

 

🧠讲解算法原理

        对于这道题,我们可以利用二分查找的思想来高效地找到山脉数组的峰顶索引。

首先,初始化左指针 left 为 1,右指针 right 为数组长度减 2。这是因为数组两端的元素不可能是峰顶(根据山脉数组的定义)。

在循环过程中,计算中间索引 mid = left + (right - left) / 2。然后比较 arr[mid] 与 arr[mid + 1] 的大小关系:

  • 如果 arr[mid] < arr[mid + 1],说明当前位置在上升坡,峰顶在 mid 的右侧,所以将 left 更新为 mid + 1
  • 如果 arr[mid] > arr[mid + 1],说明当前位置在下降坡或者已经是峰顶,峰顶在 mid 及其左侧,将 right 更新为 mid

当 left 等于 right 时,循环结束,此时 left(或 right)所指向的索引就是山脉数组的峰顶索引。

💻代码实现(以C++为例)

#include <iostream>
#include <vector>

using namespace std;

// 寻找山脉数组的峰顶索引
int peakIndexInMountainArray(vector<int>& arr) {
    int left = 1, right = arr.size() - 2;
    while (left < right) {
        int mid = left + (right - left) / 2;
        if (arr[mid] < arr[mid + 1]) {
            left = mid + 1;
        } else {
            right = mid;
        }
    }
    return left;
}

int main() {
    vector<int> arr = {0, 1, 0};
    int result = peakIndexInMountainArray(arr);
    cout << "山脉数组的峰顶索引是: " << result << endl;
    return 0;
}

复杂度分析

  • 时间复杂度:每次循环都将搜索区间缩小一半,所以时间复杂度为 O(logn),其中  是数组的长度。相比从数组头部到尾部逐个遍历查找峰顶的暴力解法(时间复杂度为O(n) ),效率有显著提升。
  • 空间复杂度:整个过程只使用了几个额外的变量来存储指针和中间索引,不需要额外的复杂数据结构,空间复杂度为O(1) ,在空间利用上非常高效。

二、寻找峰值

题目链接👉【力扣】

📖题目描述

🧠讲解算法原理

        这道题同样可以借助二分查找来解决。

初始化左指针 left 为 0,右指针 right 为数组长度减 1。

在循环中,计算中间索引 mid = left + (right - left) // 2。接着比较 nums[mid] 与 nums[mid + 1] 的大小:

  • 若 nums[mid] < nums[mid + 1],说明峰值在 mid 的右侧,将 left 更新为 mid + 1
  • 若 nums[mid] > nums[mid + 1],说明峰值在 mid 及其左侧,将 right 更新为 mid

当 left 等于 right 时,循环结束,此时返回的 left(或 right)就是一个峰值的索引。因为根据假设,数组两端虚拟的负无穷保证了一定能找到峰值。

💻代码实现(以 C++ 为例)

#include <iostream>
#include <vector>

using namespace std;

// 寻找峰值元素的索引
int findPeakElement(vector<int>& nums) {
    int left = 0, right = nums.size() - 1;
    while (left < right) {
        int mid = left + (right - left) / 2;
        if (nums[mid] < nums[mid + 1]) {
            left = mid + 1;
        } else {
            right = mid;
        }
    }
    return left;
}

int main() {
    vector<int> nums = {1, 2, 3, 1};
    int result = findPeakElement(nums);
    cout << "一个峰值元素的索引是: " << result << endl;
    return 0;
}

复杂度分析

  • 时间复杂度:由于每次迭代都能将搜索区间缩小一半,时间复杂度为 O(logn),其中 n 是数组的长度。这种方式比遍历整个数组查找峰值(时间复杂度为 )要快得多。
  • 空间复杂度:仅使用了几个简单的变量来存储指针和中间索引,没有使用额外的复杂数据结构,空间复杂度为 O(1),在空间上非常节省。

        通过对这两道题目的深入分析,我们进一步体会到二分查找算法的强大之处。在实际的算法学习和编程过程中,灵活运用二分查找及其变体,能够大大提高解决问题的效率。希望大家继续努力,不断探索算法世界的奥秘!我会持续为大家带来更多精彩的算法知识分享。

 

 

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

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

相关文章

mapbox加载geojson,鼠标移入改变颜色,设置样式以及vue中的使用

全国地图json数据下载地址 目录 html加载全部代码 方式一&#xff1a;使用html方式加载geojson 1. 初始化地图 2. 加载geojson数据 设置geojson图层样式&#xff0c;设置type加载数据类型 设置线条 鼠标移入改变颜色&#xff0c;设置图层属性&#xff0c;此处是fill-extru…

衡量算法性能的量级标准:算法复杂度

今天开始数据结构的学习&#xff01;作为一大重点&#xff0c;拿出态度很重要&#xff0c;想要真实掌握&#xff0c;博客笔记自然少不了&#xff01;重点全部上色&#xff01;避免疏忽 下面我们从0基础开始学习今天的第一节&#xff01;不用担心看不懂&#xff0c;拒绝枯燥的理…

IDE提示:因为在此系统上禁止运行脚本。有关详细信息,请参阅 https:/go.microsoft.com/fwlink/?LinkID=135170

问题情况 不知道为什么我的IDE终端运行命令的时候总提示以下内容&#xff1a; Import-Module : 无法加载文件 D:\Anaconda3\shell\condabin\Conda.psm1&#xff0c;因为在此系统上禁止运行脚本。有关详细信息&#xff0c;请参阅 https:/go.microsoft.com/fwlink/?LinkID1351…

DRF开发避坑指南01

在当今快速发展的Web开发领域&#xff0c;Django REST Framework&#xff08;DRF&#xff09;以其强大的功能和灵活性成为了众多开发者的首选。然而&#xff0c;错误的使用方法不仅会导致项目进度延误&#xff0c;还可能影响性能和安全性。本文将从我个人本身遇到的相关坑来给大…

GD32的GD库开发

所有的Cortex-M处理器都有相同的SysTick定时器&#xff0c;因为CMSIS-Core头文件中定义了一个名为SysTick的结构体。 这个定时器可以用作延时函数&#xff0c;不管是STM32的芯片还是GD32&#xff0c;AT32的芯片&#xff0c;delay函数都可以这么写&#xff0c;只要它是cortex-M…

LabVIEW项目中的工控机与普通电脑选择

工控机&#xff08;Industrial PC&#xff09;与普通电脑在硬件设计、性能要求、稳定性、环境适应性等方面存在显著差异。了解这些区别对于在LabVIEW项目中选择合适的硬件至关重要。下面将详细分析这两种设备的主要差异&#xff0c;并为LabVIEW项目中的选择提供指导。 ​ 硬件设…

python如何导出数据到excel文件

python导出数据到excel文件的方法&#xff1a; 1、调用Workbook()对象中的add_sheet()方法 wb xlwt.Workbook() ws wb.add_sheet(A Test Sheet) 2、通过add_sheet()方法中的write()函数将数据写入到excel中&#xff0c;然后使用save()函数保存excel文件 ws.write(0, 0, 1234…

Yocto项目 - 解读CROss PlatformS (CROPS)

一、概述 Yocto项目是一个用于创建自定义Linux发布版本的工具集成项目&#xff0c;在应对复杂应用场景时能提供高度可自定义性。但是在多端机应用中&#xff0c;如何在不同的平台上可靠地完成构建工作&#xff1f;CROss PlatformS (CROPS)即展示了其重要作用。 CROPS是Yocto项…

【技巧】优雅的使用 pnpm+Monorepo 单体仓库构建一个高效、灵活的多项目架构

单体仓库&#xff08;Monorepo&#xff09;搭建指南&#xff1a;从零开始 单体仓库&#xff08;Monorepo&#xff09;是一种将多个相关项目集中管理在一个仓库中的开发模式。它可以帮助开发者共享代码、统一配置&#xff0c;并简化依赖管理。本文将通过实际代码示例&#xff0…

Ubuntu24.04初始化MySQL报错 error while loading shared libraries libaio.so.1

Ubuntu24.04初始化MySQL报错 error while loading shared libraries: libaio.so.1 问题一&#xff1a;libaio1不存在 # 提示libaio1不存在 [rootzabbix-mysql-master.example.com x86_64-linux-gnu]#apt install numactl libaio1 Reading package lists... Done Building depe…

数据标注开源框架 Label Studio

数据标注开源框架 Label Studio Label Studio 是一个开源的、灵活的数据标注平台&#xff0c;旨在帮助开发者和数据科学家轻松创建高质量的训练数据集。它支持多种类型的数据&#xff08;如文本、图像、音频、视频等&#xff09;以及复杂的标注任务&#xff08;如分类、命名实体…

OS Copilot功能测评:智能助手的炫彩魔法

简介&#xff1a; OS Copilot 是一款融合了人工智能技术的智能助手&#xff0c;专为Linux系统设计&#xff0c;旨在提升系统管理和运维效率。本文详细介绍了在阿里云ECS实例上安装和体验OS Copilot的过程&#xff0c;重点评测了其三个核心参数&#xff1a;-t&#xff08;模式…

【豆包MarsCode 蛇年编程大作战】蛇形烟花

项目体验地址&#xff1a;项目体验地址 官方活动地址&#xff1a;活动地址 目录 【豆包MarsCode 蛇年编程大作战】蛇形烟花演示 引言 豆包 MarsCode介绍 项目准备 第一步&#xff1a;安装插件 第二步&#xff1a;点击豆包图标来进行使用豆包 使用豆包 MarsCodeAI助手实…

2013年蓝桥杯第四届CC++大学B组真题及代码

目录 1A&#xff1a;高斯日记&#xff08;日期计算&#xff09; 2B&#xff1a;马虎的算式&#xff08;暴力模拟&#xff09; 3C&#xff1a;第39级台阶&#xff08;dfs或dp&#xff09; 4D&#xff1a;黄金连分数&#xff08;递推大数运算&#xff09; 5E&#xff1a;前缀…

14.杂谈:领域知识库与知识图谱:概念、关系与重要性

文章目录 1. 领域知识库的概念2. 知识图谱的概念3. 领域知识库与知识图谱的关系与差异3.1 关系3.2 差异 4. 为什么要构建领域知识库&#xff1f;4.1 知识的集中管理与共享4.2 知识的标准化与规范化4.3 促进知识创新与应用 5. 为什么要进行知识融合&#xff1f;5.1 异构数据的整…

【GoLang】利用validator包实现服务端参数校验时自定义错误信息

在C/S架构下&#xff0c;服务端在校验请求参数时&#xff0c;若出现参数错误&#xff0c;要响应给客户端一个错误消息&#xff0c;通常我们会统一响应“参数错误”。 但是&#xff0c;如果只是一味的提示参数错误&#xff0c;我并不知道具体是哪个参数错了呀&#xff01;能不能…

c#实现重启Explorer.exe并且启动某个命令

由于经常需要重启Explorer.exe 然后接着又需要马上启动一个命令行&#xff0c;于是干脆写一个程序&#xff0c;实现了此功能。 可以直接在运行中&#xff0c;或者在资源管理器中新建任务。 注意&#xff0c;下方的设置为应用程序&#xff0c;可以避免启动时出现黑框。 直接上代…

C语言自定义数据类型详解(一)——结构体类型(上)

什么是自定义数据类型呢&#xff1f;顾名思义&#xff0c;就是我们用户自己定义和设置的类型。 在C语言中&#xff0c;我们的自定义数据类型一共有三种&#xff0c;它们分别是&#xff1a;结构体(struct)&#xff0c;枚举(enum)&#xff0c;联合(union)。接下来&#xff0c;我…

绘制决策树尝试2 内含添加环境变量步骤

目录 step1 ai码 ai改 step2 下面就是环境配置问题 “ExecutableNotFound: failed to execute WindowsPath(‘dot’), make sure the Graphviz executables are on your systems’ PATH” dot -v愣是没有​编辑 graphviz安装指导 对于Windows用户&#xff1a; 对于Lin…

ChatGPT被曝存在爬虫漏洞,OpenAI未公开承认

OpenAI的ChatGPT爬虫似乎能够对任意网站发起分布式拒绝服务&#xff08;DDoS&#xff09;攻击&#xff0c;而OpenAI尚未承认这一漏洞。 本月&#xff0c;德国安全研究员Benjamin Flesch通过微软的GitHub分享了一篇文章&#xff0c;解释了如何通过向ChatGPT API发送单个HTTP请求…