数据结构:时间复杂度

文章目录

  • 为什么需要时间复杂度分析?
  • 一、大O表示法:复杂度的语言
    • 1.1 什么是大O?
    • 1.2 常见复杂度速查表
  • 二、实战分析:解剖C语言代码
    • 2.1 循环结构的三重境界
      • 单层循环:线性时间
      • 双重循环:平方时间
      • 动态边界循环:隐藏的平方
    • 2.2 递归的时空折叠
      • 线性递归:阶乘计算
      • 指数递归:斐波那契噩梦
  • 三、高级技巧:复杂度组合计算
    • 3.1 顺序结构:取最大值
    • 3.2 嵌套结构:乘积法则
  • 四、常见误区与破解之道
    • 误区1:误判循环边界
    • 误区2:低估数学级数
    • 破解工具:关键公式
  • 五、复杂度优化实战
    • 案例:寻找数组中的重复元素
      • 暴力解法(O(n²))
      • 优化方案(O(n))
  • 六、自测练习
  • 结语:复杂度即格局

为什么需要时间复杂度分析?

想象你正在处理一个包含百万条数据的数组:

  • O(n²) 的算法可能需要几天才能完成
  • O(n log n) 的算法可能只需几秒
  • O(n) 的算法眨眼间就能得出结果

时间复杂度就像算法的「体检报告」,它揭示了代码执行效率如何随数据规模增长而变化。本文将用C语言示例,手把手教你掌握这项核心技能!


一、大O表示法:复杂度的语言

1.1 什么是大O?

  • 本质:描述算法执行时间的增长趋势
  • 特点:忽略常数项和低阶项,专注主要矛盾
  • 公式T(n) = O(f(n)) 表示存在常数C,使得当n足够大时,T(n) ≤ C·f(n)

1.2 常见复杂度速查表

复杂度典型场景可视化增长趋势
O(1)数组下标访问水平直线
O(log n)二分查找缓慢爬坡
O(n)遍历数组线性上升
O(n log n)快速排序优雅曲线
O(n²)冒泡排序陡峭抛物线
O(2ⁿ)暴力穷举垂直火箭

二、实战分析:解剖C语言代码

2.1 循环结构的三重境界

单层循环:线性时间

// 示例:计算数组和
int sum = 0;
for (int i = 0; i < n; i++) {  // 执行n次sum += array[i];           // O(1)操作
}
// 总复杂度:O(n)

双重循环:平方时间

// 示例:打印所有数对
for (int i = 0; i < n; i++) {       // 外层n次for (int j = 0; j < n; j++) {   // 内层n次printf("(%d,%d)", i, j);    // O(1)操作}
}
// 总复杂度:O(n) × O(n) = O(n²)

动态边界循环:隐藏的平方

for (int i = 0; i < n; i++) {       // 外层n次for (int j = 0; j < i; j++) {   // 内层i次(0到n-1)count++;                    // 总次数 = 0+1+2+...+(n-1) = n(n-1)/2}
}
// 总复杂度:O(n²)

2.2 递归的时空折叠

线性递归:阶乘计算

int factorial(int n) {if (n <= 1) return 1;          // 基准情形return n * factorial(n-1);     // 递归调用n次
}
// 调用栈深度:O(n)
// 时间复杂度:O(n)

指数递归:斐波那契噩梦

int fib(int n) {if (n <= 1) return n;return fib(n-1) + fib(n-2);    // 每次产生2个分支
}
// 时间复杂度:O(2ⁿ) (实际约为O(1.618ⁿ))

递归树呈指数级展开


三、高级技巧:复杂度组合计算

3.1 顺序结构:取最大值

void process_data(int n) {// 阶段1: O(n)for (int i=0; i<n; i++) { /* ... */ }// 阶段2: O(n²)for (int i=0; i<n; i++) {for (int j=0; j<n; j++) { /* ... */ }}
}
// 总复杂度 = O(n) + O(n²) = O(n²)

3.2 嵌套结构:乘积法则

void matrix_ops(int n) {for (int i=0; i<n; i++) {          // O(n)for (int j=1; j<n; j*=2) {     // O(log n)printf("%d", i*j);         // O(1)}}
}
// 总复杂度 = O(n) × O(log n) = O(n log n)

四、常见误区与破解之道

误区1:误判循环边界

int k = 0;
while (k < 100) {     // 固定循环100次process(data[k++]); 
}
// 真实复杂度:O(1) 而非 O(n)

误区2:低估数学级数

for (int i=1; i<=n; i*=2) {    // 执行次数:log₂nfor (int j=0; j<i; j++) {  // 内层总次数:1+2+4+...+2^log₂n = 2n-1count++;}
}
// 总复杂度:O(n) 而非 O(n log n)

破解工具:关键公式

  • 等差数列和:1+2+3+...+n = n(n+1)/2 → O(n²)
  • 等比数列和:1+2+4+...+2^k = 2^(k+1)-1 → O(2^k)
  • 对数计算:循环变量i每次乘以2 → 循环次数log₂n

五、复杂度优化实战

案例:寻找数组中的重复元素

暴力解法(O(n²))

for (int i=0; i<n; i++) {for (int j=i+1; j<n; j++) {if (arr[i] == arr[j]) return true;}
}

优化方案(O(n))

// 使用哈希表记录出现次数
int hash_table[MAX_SIZE] = {0};
for (int i=0; i<n; i++) {if (hash_table[arr[i]]++) return true;
}

六、自测练习

  1. 分析以下代码复杂度
for (int i=0; i<n; i+=5) {for (int j=0; j<1000; j++) {sum += i*j;}
}

答案:O(n)(外层循环n/5次,内层固定1000次 → 忽略常数后为线性)

  1. 递归函数复杂度分析
void fun(int n) {if (n <= 0) return;printf("%d", n);fun(n/2);fun(n/2);
}

答案:O(n)(递归树总节点数=1+2+4+…+n → 约2n个节点)


结语:复杂度即格局


不同复杂度的时间增长对比

掌握时间复杂度分析,就像获得了一副「算法透视眼镜」:

  • 在面试中快速评估解法优劣
  • 在大数据场景下避免性能灾难
  • 培养对代码的直觉敏感性

下次看到嵌套循环时,试着在心中画出它的增长曲线。当复杂度从O(n²)优化到O(n log n)时,那种思维的跃迁感,正是编程最迷人的魔法时刻!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/894545.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

S4 HANA明确税金汇差科目(OBYY)

本文主要介绍在S4 HANA OP中明确税金汇差科目(OBYY)相关设置。具体请参照如下内容&#xff1a; 1. 明确税金汇差科目(OBYY) 以上配置点定义了在外币挂账时&#xff0c;当凭证抬头汇率和税金行项目汇率不一致时&#xff0c;造成的差异金额进入哪个科目。此类情况只发生在FB60/F…

hypermesh中用tcl脚本生成多个线段

hypermesh中原本有利用多个节点生成线段的功能&#xff0c;但实际应用中不太好用&#xff0c;因为hypermesh会自动对线段进行拟合。如果手动生成多个线段&#xff0c;又太繁琐&#xff0c;这里写了一段脚本来自动生成&#xff08;#为注释符号&#xff09;。 set alist {} for …

87.(3)攻防世界 web simple_php

之前做过&#xff0c;回顾 12&#xff0c;攻防世界simple_php-CSDN博客 进入靶场 <?php // 显示当前 PHP 文件的源代码&#xff0c;方便调试或查看代码结构 // __FILE__ 是 PHP 的一个魔术常量&#xff0c;代表当前文件的完整路径和文件名 show_source(__FILE__);// 包含…

pycharm 中的 Mark Directory As 的作用是什么?

文章目录 Mark Directory As 的作用PYTHONPATH 是什么PYTHONPATH 作用注意事项 Mark Directory As 的作用 可以查看官网&#xff1a;https://www.jetbrains.com/help/pycharm/project-structure-dialog.html#-9p9rve_3 我们这里以 Mark Directory As Sources 为例进行介绍。 这…

一个类有一个全局变量 m,多线程对它进行增加操作,如何保证线程安全?

一个类有一个全局变量 m&#xff0c;多线程对它进行增加操作&#xff0c;如何保证线程安全&#xff1f; 在多线程环境下对共享变量进行修改时&#xff0c;确保线程安全的关键是保证操作的原子性、可见性和有序性。以下是针对全局变量 m 的多线程自增操作的线程安全解决方案&am…

CSS关系选择器详解

CSS关系选择器详解 学习前提什么是关系选择器&#xff1f;后代选择器&#xff08;Descendant Combinator&#xff09;语法示例注意事项 子代选择器&#xff08;Child Combinator&#xff09;语法示例注意事项 邻接兄弟选择器&#xff08;Adjacent Sibling Combinator&#xff0…

【基于SprintBoot+Mybatis+Mysql】电脑商城项目之用户注册

&#x1f9f8;安清h&#xff1a;个人主页 &#x1f3a5;个人专栏&#xff1a;【计算机网络】【Mybatis篇】 &#x1f6a6;作者简介&#xff1a;一个有趣爱睡觉的intp&#xff0c;期待和更多人分享自己所学知识的真诚大学生。 目录 &#x1f3af;项目基本介绍 &#x1f6a6;项…

Microsoft Power BI:融合 AI 的文本分析

Microsoft Power BI 是微软推出的一款功能强大的商业智能工具&#xff0c;旨在帮助用户从各种数据源中提取、分析和可视化数据&#xff0c;以支持业务决策和洞察。以下是关于 Power BI 的深度介绍&#xff1a; 1. 核心功能与特点 Power BI 提供了全面的数据分析和可视化功能&…

电控三周速成计划参考

第1周&#xff1a;基础搭建与GPIO控制 学习目标&#xff1a;建立开发环境&#xff0c;掌握最基础的硬件控制能力 每日学习&#xff08;2-3小时&#xff09;&#xff1a; 环境搭建&#xff08;2天&#xff09; 安装Keil MDK-ARM STM32CubeMX使用CubeMX创建第一个工程&#xf…

[SAP ABAP] 在ABAP Debugger调试器中设置断点

在命令框输入/H&#xff0c;点击回车以后&#xff0c;调试被激活&#xff0c;点击触发任意事件进入ABAP Debugger调试器界面 点击按钮&#xff0c;可以在Debugger调试器中新增临时断点 我们可以从ABAP命令、方法、功能、表单、异常、消息、源代码等多个维度在Debugger调试器中设…

【NEXT】网络编程——上传文件(不限于jpg/png/pdf/txt/doc等),或请求参数值是file类型时,调用在线服务接口

最近在使用华为AI平台ModelArts训练自己的图像识别模型&#xff0c;并部署了在线服务接口。供给客户端&#xff08;如&#xff1a;鸿蒙APP/元服务&#xff09;调用。 import核心能力&#xff1a; import { http } from kit.NetworkKit; import { fileIo } from kit.CoreFileK…

RssWebAll:抓取任意网页的内容生成 RSS 订阅源

RssWebAll&#xff1a;抓取任意网页的内容生成 RSS 订阅源 RssWebAll 是一个强大的工具&#xff0c;可以帮助用户抓取任意网页的内容&#xff0c;并生成相应的 RSS 订阅源&#xff0c;让用户随时随地获取他们感兴趣的内容更新。 功能亮点 简单易用&#xff1a;所见即所得&…

从一到无穷大 #43:Presto History Based Optimizer,基于PlanNode粒度统计的查询计划选择策略

本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。 本作品 (李兆龙 博文, 由 李兆龙 创作)&#xff0c;由 李兆龙 确认&#xff0c;转载请注明版权。 文章目录 引言MotivationArchitectureHBO ScenarioExperiments结束语 引言 过年回家这件事在挚…

【C++】继承(下)

大家好&#xff0c;我是苏貝&#xff0c;本篇博客带大家了解C的继承&#xff08;下&#xff09;&#xff0c;如果你觉得我写的还不错的话&#xff0c;可以给我一个赞&#x1f44d;吗&#xff0c;感谢❤️ 目录 5.继承与友元6.继承与静态成员7.复杂的菱形继承及菱形虚拟继承8.继…

项目开发实践——基于SpringBoot+Vue3实现的在线考试系统(九)(完结篇)

文章目录 一、成绩查询模块实现1、学生成绩查询功能实现1.1 页面设计1.2 前端页面实现1.3 后端功能实现2、成绩分段查询功能实现2.1 页面设计2.2 前端页面实现2.3 后端功能实现二、试卷练习模块实现三、我的分数模块实现1、 页面设计2、 前端页面实现3、 后端功能实现四、交流区…

【流媒体】搭建流媒体服务器

搭建Windows Nginx服务器 搭建 下载nginx工具包解压至本地&#xff0c;并在cmd窗口中切换至nginx所在的本地目录修改 conf/nginx.conf 文件&#xff0c;更改其端口号 server中的 listen的端口号从 80改为 8080&#xff0c;因为80经常被其他服务占用&#xff0c;导致无法打开 …

携程Java开发面试题及参考答案 (200道-下)

insert 一行数据的时候加的是什么锁?为什么? 在 MySQL 中,当执行 INSERT 操作插入一行数据时,加锁的情况会因存储引擎和具体的事务隔离级别而有所不同。一般来说,在 InnoDB 存储引擎下,INSERT 操作加的是行级排他锁(Row Exclusive Lock),以下详细说明原因。 行级排他…

洛谷P11655「FAOI-R5」Lovely 139

P11655「FAOI-R5」Lovely 139 题目背景 Update&#xff1a;数据有 0 0&#xff0c;答案为 1&#xff0c;请选手特判以正常通过。 Height ≤ 139 \text{Height}\leq139 Height≤139。 题目描述 对于一个 01 \tt 01 01 串 S S S&#xff08;下标从 1 1 1 开始&#xff09;…

【Linux】24.进程信号(1)

文章目录 1. 信号入门1.1 进程与信号的相关知识1.2 技术应用角度的信号1.3 注意1.4 信号概念1.5 信号处理常见方式概览 2. 产生信号2.1 通过终端按键产生信号2.2 调用系统函数向进程发信号2.3 由软件条件产生信号2.4 硬件异常产生信号2.5 信号保存 3. 阻塞信号3.1 信号其他相关…

《手札·开源篇》从开源到商业化:中小企业的低成本数字化转型路径 ——以Odoo为数据中台低成本实现售前售中一体化

某机电设备有限公司数字化转型案例&#xff1a;以Odoo为数据中台实现售前售中一体化 一、企业背景某机电设备有限公司在机电设备领域历经多年发展&#xff0c;业务广泛&#xff0c;涵盖工业自动化设备、电力设备等产品的销售与服务。随着业务版图不断拓展&#xff0c;企业面临…