C++二分算法的应用:寻找峰值原理、源码及测试用例

简介: C++二分算法的应用:寻找峰值原理、源码及测试用例

说明

此文是课程https://edu.csdn.net/course/detail/38771 的讲义。

源码下载:https://download.csdn.net/download/he_zhidan/88458478

本文涉及的基础知识点

二分查找算法合集

题目

长度为n的数组nums,请返回任意一峰值的索引。符合以下条件之一i便是峰值的索引。

n等于1

i等于0

n>1

i等于0

nums[i] >nums[i+1]

n>1

i等于n-1

nums[i] > nums[i-1]

0<i<n-1

nums[i]>nums[i-1]

nums[i]>nums[i+1]

题目保证nums[i]不等于nums[i+1]。

2023年11月14整补

有读者反应看不懂,感谢他的建议,特增补如下。

定义

原始数组的连续区域称为子数组。如果一个子数组的首元素在原始数组的索引为i,尾元素在原始数组的索引为j,则用nums[i,j]表示此子数组,也可以用nums[i,j+1)表示。如果nums[i,j]同时符合以下两个条件,则是好子数组。

一,i为0或nums[i]>nums[i-1]。

二,j为n-1或nums[j]>nums[j+1]

推论

一,如果好子数组nums[i,j]的长度为1,则i就是峰值。

二,如果好子数组nums[i,j]的长度大于1,则从nums[i,j]找出一个比nums[i,j]短的好子数组nums[i1,j1]。

三,由于长度不断变短,长度一定会变为1。

举例

1 2

1 2 =>2 (1不是好子数组 2是,长度降为1结束)

2 1

2  1 =>2 (2是好子数组 ,1不是,长度降为1结束)

1 2 3

1 2 3 => 2 3(1不是好子数组,2 3是)=> 3(3是好数组长度降为1结束)

3 2 1

1 2 3 => 3

1 3 2

1 3 2 => 3 2 => 3

1到4

1 2 3 4 => 3 4 => 4

1 2 4 3 => 4 3 => 4

1 3 2 4=> 1 3 => 3

1 3 4 2 => 4 2 => 4

1 4 2 3=> 1 4=> 4

1 4 3 2=> 1 4=> 4

...

2 4 1 3 => 2 4 => 4

其它

1 2 3 4 5 6 7 8 => 5 6 7 8 => 7 8 => 8

9 8 7 6 5 4 3 2 1 => 9 8 7 6 => 9 8 = > 9

1 3 5 7 9 8 6 4 2 => 9 8 6 4 2 = > 9 8 => 9

总结

将好子数组nums[i,j] 分成左右两个子数组nums[i,mid)和nums[mid,j]

如果nums[mid]> nums[mid-1] 则nums[mid,j]是好子数组

否则 nums[i,mid)是好子数组

分析

假定

nums[left,r)符合nums[left]>nums[left-1],且nums[r-1]>nums[r]。显然初始情况nums[0,n)符合。

推论一:如果[left,r)的长度为1,则left就是返回的索引。

推论二:假定left < mid<r。如果mid[mid] > mid[mid-1],则nums[mid,r)也符合假定。如果mid[mid] < mid[mid-1],则nums[left,mid)也符合假定。

推论三:推论二也可以也可以理解成分别抛弃[left,mid)和[mid,r)。令mid = left+(r-left)/2,由于r-left>=2,所以left<mid<r。也就是抛弃的子数组不会为空。也就是数组不断变短。等长度为1结束。

时间复杂度

由于每次抛弃一半,所以需要抛弃logn次。故时间复杂度O(logn)

核心代码

class Solution {
public:
    int findPeakElement(vector<int>& nums) {
        int left = 0, r = nums.size();
        while (r - left > 1)
        {
            const int mid = left + (r - left) / 2;
            if (nums[mid] > nums[mid - 1])
            {
                left = mid;
            }
            else
            {
                r = mid;
            }
        }
        return left;
    }
};

测试用例

int main()
{
    Solution slu;
    vector<int> nums = { 1,2,3,4 };
    int res = slu.findPeakElement(nums);
    assert(3 == res);
    nums = { 4,3,2,1 };
    res = slu.findPeakElement(nums);
    assert(0 == res);
    nums = { 2,5,3,1 };
     res = slu.findPeakElement(nums);
    assert(1 == res);
}

其它

学院课程

基础算法的C++实现课程,请点击下面的CSDN学院的链接。

2024年1月15之前完全免费,之后绝大部分免费

https://edu.csdn.net/course/detail/38771

C#入职培训

此课程的目的:让新同事更快完成从学生到C#程序员的转换,更快上手完成C#的开发工作。

https://edu.csdn.net/course/detail/38768

C++入职培训

让新同事更快完成从学生到C++程序员的转换,更快上手完成C++的开发工作。

https://edu.csdn.net/course/detail/32049

运行验证环境

Win10 VS2022 Ck++17 或win7 VS2019 C++17

每天都补充正能量

好好学习,天天向上。

事无终始,无务多业。

是故置本不安者,无务丰末。

相关下载

如果你时间宝贵,只想看精华,请到CSDN下载频道下载《闻缺陷则喜算法册》doc版

https://download.csdn.net/download/he_zhidan/88348653


相关文章
|
1天前
|
自然语言处理 算法 搜索推荐
分词算法的基本原理及应用
分词算法的基本原理及应用
|
3天前
|
存储 算法 安全
深入解析RSA算法原理及其安全性机制
深入解析RSA算法原理及其安全性机制
|
3天前
|
存储 算法 安全
MD5哈希算法:原理、应用与安全性深入解析
MD5哈希算法:原理、应用与安全性深入解析
|
3天前
|
算法 安全 Java
AES加解密算法:原理、应用与安全性解析
AES加解密算法:原理、应用与安全性解析
|
3天前
|
机器学习/深度学习 并行计算 算法
技术经验解读:《人工神经网络》第9章遗传算法原理
技术经验解读:《人工神经网络》第9章遗传算法原理
|
安全 Java 测试技术
python接口自动化(三)--如何设计接口测试用例(详解)
上篇我们已经介绍了什么是接口测试和接口测试的意义。在开始接口测试之前,我们来想一下,如何进行接口测试的准备工作。或者说,接口测试的流程是什么?有些人就很好奇,接口测试要流程干嘛?不就是拿着接口文档直接利用接口 测试工具测试嘛。其实,如果只是三五个接口,你可以这么做一个临时的接口测试。但是,如果是上百个接口,或者,你们公司的这个项目,第一次做接口测试,那么,我们还是很有必要严格遵守接口测试的流程。
314 0
python接口自动化(三)--如何设计接口测试用例(详解)
|
测试技术
正交试验测试用例设计及工具推荐
在科研和生产实践中,人们往往要做许多次实验来进行某项研究。实验条件一般包括很多因素,当因素的值不同时,实验的结果也不一样。如果想把每个因素的每个值都要实验一遍,总实验数就等于各因素的值的个数的乘积,而这个数往往很大,超过了可接受的成本。 例如,假设某个实验由A,B,C,D四个因素,每个因素都有10个不同的取值,那么如果想把每个因素都考虑到,我们需要做 10*10*10*10=10000次实验。 为了减少实验数目,我们必须选出那些最有代表性的例子。于是,就要用到了正交表法(Orthogonal Array Testing Strategy)。
275 0
正交试验测试用例设计及工具推荐
|
测试技术 数据库 数据安全/隐私保护
测试用例设计之业务流程分析法
测试用例设计之业务流程分析法
227 0
测试用例设计之业务流程分析法
|
算法 Java 测试技术
边界值分析法测试用例设计实例
边界值分析法是黑盒测试的重要方法,本文以一道数位DP算法题为例,自主测试黑盒测试用例,并采用JUnit5完成单元测试。
156 0
|
算法 安全 测试技术
【软件测试】测试用例的设计方法
测试用例写的过于简单,则可能失去了测试用例的意义,设计过于简单的测试用例其实并没有真正的进行设计,只是把需要测试的功能模块记录下来而已,它的作用仅仅是在测试过程中作为一个简单的测试计划,提醒测试人员测试的主要功能包括哪些而已,测试用例设计的本质应该是在设计的过程中理解需求,检验需求,并把对软件系统的测试方法的思路记录下来,以便指导将来的测试
【软件测试】测试用例的设计方法