拜托,面试别再问我最大值最小值了!!!

简介: 如何从n个数里找到最大值?很容易想到,用一个循环就能搞定。

如何从n个数里找到最大值?

很容易想到,用一个循环就能搞定。

int find_max(int arr[n]){

    int max = -infinite;

    for(int i=0; i<n; i++)

        if(arr[i]>max)

            max=arr[i];

    return max;

}

这里,需要执行n-1次比较。

如何从n个数里找到最大值与最小值?

很容易想到,用一个循环找到最大值和最小值,就能搞定。

(int, int) find_max_min(int arr[n]){

    int max = -infinite;

    int min = infinite;

 

    for(int i=0; i<n; i++){

        if(arr[i]>max)

            max=arr[i];

        if(arr[i]<min)

            min=arr[i];

    }

 

    return (max, min);

}

这里,需要执行2*(n-1)=2n-2次比较。

还有没有更快的方法呢?

分治法或许可以派上用场,分治法的思路是:

(1)把大规模拆分成小规模;

(2)小规模分别求解;

(3)小规模求解之后,再综合求解大规模;

看能不能往这个例子里套用:

(1)将arr[0,n]分为arr[0,n/2]和arr[n/2,n];

(2)每个子数组分别求解最大值和最小值;

(3)两个子数组的最大值里再取最大值,两个子数组的最小值里再取最小值,就是最终解;

伪代码大概是这样:

(int, int) find_max_min(int arr[0,n]){

    // 递归左半区

    (max1, min1) = find_max_min(arr[0, n/2]);

    // 递归右半区

    (max2, min2) = find_max_min(arr[n/2, n]);

 

    // 再计算两次

    max = max1>max2?max1:max2;

    min = min1<min2?min1:min2;

 

    return (max, min);

}

画外音,实际的递归代码要注意:

(1)入参不是0和n,而是数组的下限和上限;

(2)递归要收敛,当数组的上下限相差1时,只比较一次,直接返回max和min,而不用再次递归;

分治法之后,时间复杂度是多少呢?

如果你是“架构师之路”的老读者,《搞定所有时间复杂度计算》一文,能够轻松求解分治法的时间复杂度分析:

当只有2个元素时,只需要1次计算就能知道最大,最小值

当有n个元素时,

 (1)递归左半区;

 (2)递归右半区;

 (3)再进行两次计算;

f(2)=1;【式子A】

f(n)=2*f(n/2)+2;【式子B】

求解递归式子,得到:

f(n)=1.5n-2;

画外音,证明过程如下:

【式子B】不断展开能得到:

f(n)=2*f(n/2)+2;【式子1】

f(n/2)=2*f(n/4)+2;【式子2】

f(n/4)=2*f(n/8)+2;【式子3】

...

f(n/2^(m-1))=2*f(2^m)+2;【式子m】

通过这m个式子的不断代入,得到:

f(n)=(2^m)*f(n/2^m)+2^(m+1)-2;【式子C】

由于f(2)=1【式子A】;

即【式子C】中n/2^m=2时,f(n/2^m)=f(2)=1;

此时n=2^(m+1),代入【式子C】

即f(n)=n/2 + n -2 = 1.5n-2;

证明过程很严谨,但我知道你没看懂。

建议再看看《搞定所有时间复杂度计算》。

总结,n个数:

  • 求最大值,遍历,需要n-1次计算
  • 求最大最小值,遍历,需要2n-2次计算
  • 求最大最小值,分治,时间复杂度1.5n-2

画外音:别跳出,文末有作业。

思路比结论重要,希望大家有收获。

image.png
架构师之路-分享可落地的技术文章

目录
相关文章
|
Java API Spring
Spring的设计哲学--来自官方
Spring框架设计哲学强调在每个层级提供选择,允许延迟设计决策,如通过配置切换持久性提供商。它拥抱灵活性,适应不同观点,同时保持强向后兼容性,确保版本间少有破坏性更改。Spring注重API设计,追求高质量代码,拥有清晰无循环依赖的结构。这些原则使Spring成为Java开发中最受欢迎的框架之一。
|
前端开发 安全 JavaScript
NFT数字藏品平台的基本原理和开发过程
NFT 数字藏品平台是一个数字化的收藏品交易平台,使收藏品的交易和管理更加方便和透明。为了实现数字藏品平台的功能,需要有一个稳定、可靠、高效的技术架构。本文将介绍数字藏品平台开发的技术架构,以便了解 NFT数字藏品平台的基本原理和开发过程。
|
人工智能 开发框架 Java
重磅发布!AI 驱动的 Java 开发框架:Spring AI Alibaba
随着生成式 AI 的快速发展,基于 AI 开发框架构建 AI 应用的诉求迅速增长,涌现出了包括 LangChain、LlamaIndex 等开发框架,但大部分框架只提供了 Python 语言的实现。但这些开发框架对于国内习惯了 Spring 开发范式的 Java 开发者而言,并非十分友好和丝滑。因此,我们基于 Spring AI 发布并快速演进 Spring AI Alibaba,通过提供一种方便的 API 抽象,帮助 Java 开发者简化 AI 应用的开发。同时,提供了完整的开源配套,包括可观测、网关、消息队列、配置中心等。
7160 120
|
12月前
|
存储 关系型数据库 MySQL
MySQL主从复制原理和使用
本文介绍了MySQL主从复制的基本概念、原理及其实现方法,详细讲解了一主两从的架构设计,以及三种常见的复制模式(全同步、异步、半同步)的特点与适用场景。此外,文章还提供了Spring Boot环境下配置主从复制的具体代码示例,包括数据源配置、上下文切换、路由实现及切面编程等内容,帮助读者理解如何在实际项目中实现数据库的读写分离。
1216 1
MySQL主从复制原理和使用
|
消息中间件 存储 大数据
聊一聊几款主流消息队列之间的差异,我们应该如何选择
聊一聊几款主流消息队列之间的差异,我们应该如何选择
745 2
|
11月前
|
监控 API 数据安全/隐私保护
小红书详情API接口的获取与应用
在互联网信息爆炸的时代,小红书凭借丰富的用户生成内容(UGC)和精准的推荐系统迅速崛起,成为重要的社区电商平台。为了帮助开发者高效利用平台数据,小红书开放平台提供了多种API接口,涵盖商品详情和笔记详情等。本文详细介绍了如何注册、申请权限、构建请求、处理响应及应用这些API接口,旨在为开发者提供全面的指南,助力数据驱动的决策与创新。
4592 1
|
Java Spring
基于Java多线程处理数据
【4月更文挑战第9天】 基于Java多线程处理数据
|
SQL 关系型数据库 MySQL
(十八)MySQL排查篇:该如何定位并解决线上突发的Bug与疑难杂症?
前面《MySQL优化篇》、《SQL优化篇》两章中,聊到了关于数据库性能优化的话题,而本文则再来聊一聊关于MySQL线上排查方面的话题。线上排查、性能优化等内容是面试过程中的“常客”,而对于线上遇到的“疑难杂症”,需要通过理性的思维去分析问题、排查问题、定位问题,最后再着手解决问题,同时,如果解决掉所遇到的问题或瓶颈后,也可以在能力范围之内尝试最优解以及适当考虑拓展性。
1042 3
|
缓存 监控 安全
Spring AOP 详细深入讲解+代码示例
Spring AOP(Aspect-Oriented Programming)是Spring框架提供的一种面向切面编程的技术。它通过将横切关注点(例如日志记录、事务管理、安全性检查等)从主业务逻辑代码中分离出来,以模块化的方式实现对这些关注点的管理和重用。 在Spring AOP中,切面(Aspect)是一个模块化的关注点,它可以跨越多个对象,例如日志记录、事务管理等。切面通过定义切点(Pointcut)和增强(Advice)来介入目标对象的方法执行过程。 切点是一个表达式,用于匹配目标对象的一组方法,在这些方法执行时切面会被触发。增强则定义了切面在目标对象方法执行前、执行后或抛出异常时所
16633 4