• 数据结构与算法——算法时间复杂度


    ced485cbb11e458d81a746890b32cf3f.gif

     作者:敲代码の流川枫

    博客主页:流川枫的博客

    专栏:和我一起学java

    语录:Stay hungry stay foolish

    工欲善其事必先利其器,给大家介绍一款超牛的斩获大厂offer利器——牛客网

    点击免费注册和我一起刷题吧     

    文章目录

    1. 大O记法

    2. 推导大O阶方法

    3. 常数阶

    4. 线性阶

    5. 对数阶

    6. 平方阶

    7. 分析时间复杂度

    7.1func1的时间复杂度

    7.2 func2的时间复杂度 

    7.3 binarySearch的时间复杂度 

     7.4 阶乘递归factorial的时间复杂度

    7.5 斐波那契递归fibonacci的时间复杂度 

    7.6 bubbleSort的时间复杂度


    1. 大O记法

    判断一个算法好不好,只通过少量的数据是不能做出准确判断的

    T(n)= O(f(n)

    在进行算法分析时,语句总的执行次数T(n)是关于问题规模n的函数,进而分析T(n)随n的变化情况并确定T(n)的数量级。算法的时间复杂度就是算法的时间量度,记为T(n)= O(f(n))

    它表示某个算法,随着问题规模n的增大,算法执行时间的增长率和f(n)的增长率相同,称作算法的渐近时间复杂度,简称时间复杂度。f(n)为问题规模n的某个函数

    这样用O()来体现算法时间复杂度的记法称为大O记法

    显然,在一般情况下,随着n的增大,T(n)增长最慢的算法为最优算法

    2. 推导大O阶方法

    1.用常数1取代运行时间中的加法常数

    2.修改后的运行次数函数中,只保留最高阶项

    3.去掉最高阶项的系数(如果系数不为1)

    这样就得到了大O阶

    接下来是常见的大O阶

    3. 常数阶

    1. //执行一次
    2. int sum = 0,n=10;
    3. //执行一次
    4. sum =1+n;
    5. //执行一次
    6. System.out.println(sum);

    运行次数f(n)=3

    根据我们的推导方法,第一步就是把常改为1,在保留最高阶项发现它没有最高阶项,所以它的时间复杂度是O(1)

    再者,如果sum=1+n;这个语句有十句,则会执行12次。事实上,无论n为多少,两者的差异就是3次和12次,与n的大小是无关的,不会随着n的变大而发生变化,执行时间恒定的算法,我们称为具有O(1)的时间复杂度,又叫常数阶

    4. 线性阶

    线性阶的循环结构会复杂很多。关键是分析循环结构的运行情况

    1. for (int i = 0; i < n; i++) {
    2. // 时间复杂度为O(1)的程序步骤序列
    3. }

    循环时间复杂度为O(n),因为循环体中的代码要执行n次

    5. 对数阶

    1. int count = 1;
    2. while(count <n){
    3. count = count*2;
    4. //时间复杂度为O(1)的程序步骤序列
    5. }

    由于每次count*2以后,就距离n更近了,当多少个2相乘以后大于n,就会退出循环

    由2^x=n得到x=log2(n),所以这个循环的时间复杂度为O(logn) 

    6. 平方阶

     看一个循环嵌套

    1. for (int i = 0; i < n; i++) {
    2. for (int j = 0; j < n; j++) {
    3. //时间复杂度为O(1)的程序步骤序列
    4. }
    5. }

     内循环为O(n),对于外循环,不过是内部时间复杂度为O(n)的语句在循环n次,所以时间复杂度为O(n^2)

    如果循环次数改为m,时间复杂度就变为O(m*n)

    7. 分析时间复杂度

    下面我们看几段代码分析一下时间复杂度

    7.1func1的时间复杂度

    1. // 计算func1的时间复杂度?
    2. void func1(int N) {
    3. int count = 0;
    4. for (int k = 0; k < 2 * N ; k++) {
    5. count++;
    6. }
    7. int M = 10;
    8. while ((M--) > 0) {
    9. count++;
    10. }
    11. System.out.println(count);
    12. }

     基本操作执行了2N+10次,通过推导大O阶方法知道,时间复杂度为 O(N)

    7.2 func2的时间复杂度 

    1. // 计算func2的时间复杂度?
    2. void func2(int N) {
    3. int count = 0;
    4. for (int k = 0; k < 100; k++) {
    5. count++;
    6. }
    7. System.out.println(count);
    8. }

    基本操作执行了100次,通过推导大O阶方法,时间复杂度为 O(1) 

    7.3 binarySearch的时间复杂度 

    1. // 计算binarySearch的时间复杂度?
    2. int binarySearch(int[] array, int value) {
    3. int begin = 0;
    4. int end = array.length - 1;
    5. while (begin <= end) {
    6. int mid = begin + ((end-begin) / 2);
    7. if (array[mid] < value)
    8. begin = mid + 1;
    9. else if (array[mid] > value)
    10. end = mid - 1;
    11. else
    12. return mid;
    13. }
    14. return -1;
    15. }

    基本操作执行最好1次,最坏 log2(n)次,时间复杂度为 O(log2(n)) ps: og2(n)在算法分析中表示是底数 为2,对数为N,有些地方会写成lgN。(因为二分查 找每次排除掉一半的不适合值,一次二分剩下:n/2 两次二分剩下:n/2/2 = n/4)。即n/(2^x),x=logn

     7.4 阶乘递归factorial的时间复杂度

    1. // 计算阶乘递归factorial的时间复杂度?
    2. long factorial(int N) {
    3. return N < 2 ? N : factorial(N-1) * N;
    4. }

    递归的时间复杂度=递归次数*每次递归之后执行的次数

    基本操作递归了N次,时间复杂度为O(N) 

    7.5 斐波那契递归fibonacci的时间复杂度 

    1. // 计算斐波那契递归fibonacci的时间复杂度?
    2. int fibonacci(int N) {
    3. return N < 2 ? N : fibonacci(N-1)+fibonacci(N-2);
    4. }

    参考一下斐波那契数的递归图 

    分析发现基本操作递归了2^n 次,时间复杂度为O( 2^n) 

    7.6 bubbleSort的时间复杂度

    1. // 计算bubbleSort的时间复杂度?
    2. void bubbleSort(int[] array) {
    3. for (int end = array.length; end > 0; end--) {
    4. boolean sorted = true;
    5. for (int i = 1; i < end; i++) {
    6. if (array[i - 1] > array[i]) {
    7. Swap(array, i - 1, i);
    8. sorted = false;
    9. }
    10. }
    11. if (sorted == true) {
    12. break;
    13. }
    14. }
    15. }

    基本操作执行最好N次,最坏执行了(N*(N-1))/2次,通过推导大O阶方法+时间复杂度一般看最坏,时间 复杂度为 O(N^2)

    “ 本期的分享就到这里了, 记得给博主一个三连哈,你的支持是我创作的最大动力!

    ced485cbb11e458d81a746890b32cf3f.gif

  • 相关阅读:
    C#远程调试
    21天打卡挑战 - 经典算法之快速排序
    python-(6-3-2)爬虫---requests入门(基于post请求)
    Redis的学习
    成功解决ModuleNotFoundError: No module named ‘sklearn.ensemble.weight_boosting‘
    MYSQL存储过程注释详解
    Rust 中级教程 第5课——trait(3)
    数仓:数仓建设中的数据建模和日志体系
    接口流量突增,如何做好性能调优?
    com.upd.sso.client.SsoFilter.init(SsoFilter.java..)无法访问swagger-ui.html
  • 原文地址:https://blog.csdn.net/chenchenchencl/article/details/126802871