算法复杂度精讲——算法时间复杂度的数学原理:从O(n(log(n))说起

2024-04-22 13:58

本文主要是介绍算法复杂度精讲——算法时间复杂度的数学原理:从O(n(log(n))说起,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述:在设计算法的时候,要考虑两个方面,一个是算法的正确性,另外一个就是算法的效率,也就是复杂度,通常情况下,我们优先考虑的是时间复杂度,这也是本文要讨论的内容。算法学习的时候,经常碰到这样的问题,为什么快速排序的时间复杂度是O(nlog(n))?为何插入排序的时间复杂度是On^2)?这些是我们熟悉的算法时间复杂度,可能病没有太大的问题,那我们不熟悉的呢?如果我们采用三路归并排序而不是二路归并排序,时间复杂度是多少呢?一个排序算法经过某种变形以后时间复杂度又是多少呢?本文,主要从数学底层,讲述一个算法时间复杂度是如何推导的。让你真正知其所以然,而不仅仅是总是心中存有疑惑:为何快排的时间复杂度会是这么奇怪的Onlogn))


1.数学基础知识


首先,介绍以下数学基础知识,这些基本都分布在高等数学和离散数学之中,不进行数学推导。

一些不等式:

  






无论是归并还是快速排序,我们都可以把它们归结到递归/分治这一类问题的求解,他们具有一个一般性的时间复杂度表述:


这个等式的意义是:规模是n的问题可以拆分成a个规模是n/b的问题,那么它的时间复杂度就等于a个规模是n/b的问题,加上一次分解耗费的时间Dn)和一次合并耗费的时间Cn)。第二部分到第四部分将介绍三种求解这个方程式的方法。


2.递归树方法


这是一种最直观的方法,它把上述等式形象化,然后进行求解,我们通过一个例子来说明这个情况。

例子:利用递归树求解T(n)=T(n/10)+T(9n/10)+cn

划出递归树如下:



关键点:求出树的深度和每层的代价(注意,此例中因为每层的代价都相同,所以比较好求解;但在其他情况下,可能是每层代价不同,而是一个等比数列或者其他形式的数列)


1)其中,树的深度容易求解:n-->9/10n-->81/100(n)......n/n=1

也就是说,这个递归下降满足这个趋势(其中b=10/9):

于是

所以



2)第i层的代价:

每层的规模分别是1/10n9/10n,而每个节点的代价是cn/109cn/10,所以加在一块是cn


3)总代价:


所以

Tn=Onlgn


3.主方法


形如下列表达式的算法复杂度表述

Tn=aTn/b+fn)(a>=1,b>1

主方法的证明:参考算法导论第四章

最终利用等比数列的求和公式即可求解。


4.替换代换法


说明:此种方法需要凭借一定的经验,有点类似于数学归纳法,先猜测后证明。

1)步骤:猜测时间复杂度的表述形似

2)要点:猜测要准确,归纳假设要足够强,避免弱化证明。替换非多项式变量

对于边界问题:可采用移动边界和强化归纳假设的方式加以解决。


实例:

1)证明Tn=Tn/2+n的时间复杂度位Onlogn

Tn<cnlgn


说明:此种算法复杂度的计算对以分支法为基础的算法比较有效。

这篇关于算法复杂度精讲——算法时间复杂度的数学原理:从O(n(log(n))说起的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/925968

相关文章

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

golang 日志log与logrus示例详解

《golang日志log与logrus示例详解》log是Go语言标准库中一个简单的日志库,本文给大家介绍golang日志log与logrus示例详解,感兴趣的朋友一起看看吧... 目录一、Go 标准库 log 详解1. 功能特点2. 常用函数3. 示例代码4. 优势和局限二、第三方库 logrus 详解1.

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

Python如何获取域名的SSL证书信息和到期时间

《Python如何获取域名的SSL证书信息和到期时间》在当今互联网时代,SSL证书的重要性不言而喻,它不仅为用户提供了安全的连接,还能提高网站的搜索引擎排名,那我们怎么才能通过Python获取域名的S... 目录了解SSL证书的基本概念使用python库来抓取SSL证书信息安装必要的库编写获取SSL证书信息

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

golang字符串匹配算法解读

《golang字符串匹配算法解读》文章介绍了字符串匹配算法的原理,特别是Knuth-Morris-Pratt(KMP)算法,该算法通过构建模式串的前缀表来减少匹配时的不必要的字符比较,从而提高效率,在... 目录简介KMP实现代码总结简介字符串匹配算法主要用于在一个较长的文本串中查找一个较短的字符串(称为

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J