求最大最小值,最少要进行多少次比较? | 经典面试题

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

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

int find_max(int arr[n]){

    int max = -infinite;

    for(int i=0; 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

        if(arr[i]>max)

            max=arr[i];

        if(arr[i]

            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

 

    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

 

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

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

作业题,n个数:

(1)求最大值,n-1次计算,是最快的方法吗?

(2)求最大最小值,1.5n-2,是最快的方法吗?

发布者:糖太宗,转载请注明出处:https://www.qztxs.com/archives/science/technology/6014

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022年5月11日 下午8:15
下一篇 2022年5月11日 下午8:49

相关推荐

  • 数据库索引,终于懂了

    不少朋友留言问MySQL索引底层的实现,让我讲讲B+树。知其然,知其所以然,讲懂B+树其实不难,今天更多聊聊“数据库索引,为什么设计成这样”。   问题1. 数据库为什么要设计索引?   图书馆存了1000W本图书,要从中找到《架构师之路》,一本本查,要查到什么时候去? 于是,图书管理员设计了一套规则: (1)一楼放历史类,二楼放文学类,三楼放IT类… (2...

    2022年5月14日
    2700
  • MySQL官方的数据库中间件,有人用么?

    MySQL官方的数据库中间件,mysql-proxy,有童鞋了解么?   mysql-proxy是什么? mysql-proxy是mysql官方提供的mysql中间件服务,上游可接入若干个mysql-client,后端可连接若干个mysql-server。 画外音:中间件有基于客户端的,也有基于服务端的,此为后者。   mysql-proxy使用什么协议? ...

    2022年5月11日
    3100
  • crontab误删除恢复

    事故原因分析: 回忆自己操作过程中,未进行crontab的清空,网上查了下原因,并且复现了下。可能原因如下: 如果在SSH远程终端中敲下“crontab”命令之后,远程连接被一些原因(比如 糟糕的网络,程序异常)意外终止了,那么Crontab计划任务就会被操作系统所清空。听起来很不可思议,但是经过在虚拟机上的多次测试,它确确实实的发生了。测试方式为 用Sec...

    技术 2022年6月13日
    2500
  • Git修改远程URL

    Github 设置使用了ssh key,每次提交代码还是需要输入用户名???   检查发现,在设置远程url的时候使用了https https://github.com/Pa55w0rd/pa55w0rd.github.io.git 使用git remote set-url命令从https切换到ssh 1 # git remote set-url ...

    技术 2022年5月28日
    3500
  • OpenRASP 单机版

    0x00 前言 某web应用一直被黑,加上是零几年的网站代码没人维护,于是从某云开个服务器放上去了,然后再部署一个防护软件,想装个安全狗的,听大佬说存在数据回传,之前关注了百度开源的OpenRASP项目,先在测试环境试一下   0x01 环境 这个PHP测试环境是前几天CVE-2019-11043 PHP-FPM + Nginx RCE的漏洞靶场 ...

    2022年6月13日
    5800

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信