分析归并排序算法的时间复杂度,可以根据算法的逻辑,分析每一个步骤的最坏情况,然后得到总体的时间复杂度;也可以利用数学中的『归纳假设法』,用几乎纯数学的方式来得到它的时间复杂度。而后者比前者好理解得多,所以,我认为要推导归并排序的时间复杂度的话,归纳假设的方法是不二之选。
『1』时间复杂度递推公式
将待排序的数组记为a。假设这里是用递归的方式实现的归并排序(易于理解)。
归并排序将a分为两半,分别排序,再把排序后的这两半归并为一个数组。对这两半,还是用归并排序的方法,把它们分别再分成两半来排序、归并。这样一直拆分下去,直到最后拆成的“两半”已经是两个单独的元素,就算是“到头”了(没得拆了)。
因此,在每一次递归的过程中,需要做两个递归调用(分别排序拆成的那“两半”数组),以及一个归并操作。
文章来源:http://www.codelast.com/
归并操作的时间复杂度是 ,而两个递归调用的时间复杂度是多少呢?
这里先假设一次递归调用的过程所耗费的时间是 ,则在问题的规模减半后(数组拆成了两半,所以规模减半了),递归调用耗费的时间就是
了。
因此,归并排序耗费的时间的递推公式就是:
根据这个式子,我们如何用归纳假设法,求出归并排序的时间复杂度?
文章来源:http://www.codelast.com/
『2』归纳
归纳就是总结出归并排序的时间复杂度公式,说白了根据几个已知的计算结果来猜。
根据前面的递推公式,我们可以计算出:
data:image/s3,"s3://crabby-images/95d79/95d79cc4fad56dd2e931fad626eaf6fd45d1704b" alt="t(2) = 2t(1) + 2 = 2 = 2 \cdot {\log _2}2"
data:image/s3,"s3://crabby-images/b91f4/b91f4d28249381e10e96761bc89ef404e19fcb85" alt="t(4) = 2t(2) + 4 = 8 = 4 \cdot {\log _2}4"
data:image/s3,"s3://crabby-images/563f6/563f67e2e86c6b7b02a47a13066e9d044b2b8fdc" alt="t(8) = 2t(4) + 8 = 24 = 8 \cdot {\log _2}8"
计算到这里,规律已经非常明显了:
data:image/s3,"s3://crabby-images/8337d/8337d082e6050ba6ed6eae1e05491c13d968932b" alt="t(n) = n{\log _2}n"
我们知道,在描述算法复杂度的时候,对数的底数无关紧要,通常省略它,因此这里我们就记为:
data:image/s3,"s3://crabby-images/0c3e7/0c3e7308eabb296d53ebdcb20a3af141261b0345" alt="t(n) = n\log n"
所以我们就归纳(也就是猜测)出了归并排序的时间复杂度。
文章来源:http://www.codelast.com/
『3』假设 & 证明
假设什么呢?
我们当然是要假设上面归纳出的时间复杂度公式是正确的。在这样的假设下,来看看可以怎样证明归并排序的时间复杂度确实是
data:image/s3,"s3://crabby-images/e1b36/e1b3691dfd528d7119d59b21281ccaa64d8f1f7c" alt="O\left( {n\log n} \right)"
由算法效率的定义,若一个算法的时间复杂度为
data:image/s3,"s3://crabby-images/e1b36/e1b3691dfd528d7119d59b21281ccaa64d8f1f7c" alt="O\left( {n\log n} \right)"
data:image/s3,"s3://crabby-images/e90f8/e90f80d9e39ff705425dee6ccca87d17ba2e3a29" alt="c"
data:image/s3,"s3://crabby-images/f886e/f886e4312bb2494ede219407c98a4e3914868af2" alt="N"
data:image/s3,"s3://crabby-images/62c6a/62c6aded7acbfacc637dee3fe29c00c1406d0beb" alt="f(n) \le c\left( {n\log n} \right)"
data:image/s3,"s3://crabby-images/3cda8/3cda8df5299c1e736532490b12d43b7e7fab2d0d" alt="n \ge N"
![t(n) = 2t\left( {\frac{n}{2}} \right) + n \le c \cdot \left[ {\frac{n}{2}\log \left( {\frac{n}{2}} \right)} \right] + n](https://www.codelast.com/wp-content/plugins/latex/cache/tex_996b100e709f1e52518379bf8952497b.gif)
在这里,我们根据假设的正确的时间复杂度公式,用算法效率的定义,得到了上面的不等式,我们下面要做的,就是继续推导这个不等式,使之满足:
data:image/s3,"s3://crabby-images/d45e5/d45e5bdd693118a13c83af0f0a3a25b6d29e4b89" alt="t(n) \le c\cdot\left( {n\log n} \right)"
data:image/s3,"s3://crabby-images/e1b36/e1b3691dfd528d7119d59b21281ccaa64d8f1f7c" alt="O\left( {n\log n} \right)"
文章来源:http://www.codelast.com/
看推导:
![\begin{array}{l}t(n) = 2t\left( {\frac{n}{2}} \right) + n \le c \cdot \left[ {\frac{n}{2}\log \left( {\frac{n}{2}} \right)} \right] + n \le cn\log \left( {\frac{n}{2}} \right) + n\\ = cn(\log n - \log 2) + n = cn\log n - cn + n = cn\log n + (1 - c)n\end{array}](https://www.codelast.com/wp-content/plugins/latex/cache/tex_521e5bc1d53ac7e219b73396afadf3a7.gif)
此时,我们只要取
data:image/s3,"s3://crabby-images/dfe5c/dfe5c0dfbe318c423d9530e453d4965b515c2aa3" alt="c \ge 1"
data:image/s3,"s3://crabby-images/a8b57/a8b579ef4b5be92fe78153de6aa89f28a3df54a3" alt="\le cn\log n"
data:image/s3,"s3://crabby-images/c69b7/c69b7f5a0e8c02ae2ff986f8b3368ce19faa1aa4" alt="t(n) \le c \cdot \left( {n\log n} \right)"
所以到这里,我们就成功地利用假设,证明了归并排序的时间复杂度确实是
data:image/s3,"s3://crabby-images/e1b36/e1b3691dfd528d7119d59b21281ccaa64d8f1f7c" alt="O\left( {n\log n} \right)"
文章来源:https://www.codelast.com/
➤➤ 版权声明 ➤➤
转载需注明出处:codelast.com
感谢关注我的微信公众号(微信扫一扫):