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

简介: 如何从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)=2f(n/2)+2;【式子B】

 

求解递归式子,得到:

f(n)=1.5n-2;

 

画外音,证明过程如下:


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

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

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

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

...

f(n/2^(m-1))=2f(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

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

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

本文转自“架构师之路”公众号,58沈剑提供。

目录
相关文章
【牛客面试必刷TOP101】有效括号序列、滑动窗口的最大值
【牛客面试必刷TOP101】有效括号序列、滑动窗口的最大值
|
机器学习/深度学习 人工智能 算法
[leetcode/lintcode 题解] 算法面试真题详解:最大值在界内的子数组个数
[leetcode/lintcode 题解] 算法面试真题详解:最大值在界内的子数组个数
[leetcode/lintcode 题解] 算法面试真题详解:最大值在界内的子数组个数
大厂面试真题详解:山脉序列中的最大值
大厂面试真题详解:山脉序列中的最大值
大厂面试真题详解:山脉序列中的最大值
|
3月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
9天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
11天前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
35 4
|
1月前
|
算法 Java 数据中心
探讨面试常见问题雪花算法、时钟回拨问题,java中优雅的实现方式
【10月更文挑战第2天】在大数据量系统中,分布式ID生成是一个关键问题。为了保证在分布式环境下生成的ID唯一、有序且高效,业界提出了多种解决方案,其中雪花算法(Snowflake Algorithm)是一种广泛应用的分布式ID生成算法。本文将详细介绍雪花算法的原理、实现及其处理时钟回拨问题的方法,并提供Java代码示例。
67 2
|
1月前
|
JSON 安全 前端开发
第二次面试总结 - 宏汉科技 - Java后端开发
本文是作者对宏汉科技Java后端开发岗位的第二次面试总结,面试结果不理想,主要原因是Java基础知识掌握不牢固,文章详细列出了面试中被问到的技术问题及答案,包括字符串相关函数、抽象类与接口的区别、Java创建线程池的方式、回调函数、函数式接口、反射以及Java中的集合等。
28 0
|
3月前
|
XML 存储 JSON
【IO面试题 六】、 除了Java自带的序列化之外,你还了解哪些序列化工具?
除了Java自带的序列化,常见的序列化工具还包括JSON(如jackson、gson、fastjson)、Protobuf、Thrift和Avro,各具特点,适用于不同的应用场景和性能需求。
|
3月前
|
Java
【Java基础面试三十七】、说一说Java的异常机制
这篇文章介绍了Java异常机制的三个主要方面:异常处理(使用try、catch、finally语句)、抛出异常(使用throw和throws关键字)、以及异常跟踪栈(异常传播和程序终止时的栈信息输出)。