📌 一、 线性表(Linear List)的核心特征与物理存储
在计算机科学中,线性表 是最基础且应用最广泛的数据结构之一。根据物理存储方式的不同,线性表主要分为顺序表 和链表。
1. 逻辑物理架构对比
- 逻辑特征 :具有唯一前驱 、唯一后继(首尾元素除外)。
- 物理存储分类 :
- 连续存放 →\rightarrow→ 对应 顺序表
- 分散存放 →\rightarrow→ 对应 链表
💡 泛型说明 :在结构体或算法定义中,我们通常使用
ElemType来代表任意类型。
2. 【核心方法总结】顺序表 vs 链表 性能博弈
在 408 选择题和算法大题中,如何根据场景选择存储结构是核心考点:
- 顺序表 :支持随机存取 (即可以通过下标在 O(1)O(1)O(1) 时间内直接获取目标元素),但是其插入和删除操作较为困难 (需要移动大量的元素,平均时间复杂度为 O(n)O(n)O(n))。
- 链表 :不支持随机存取 (访问第 iii 个元素需要从头节点开始遍历,时间复杂度为 O(n)O(n)O(n)),但是其插入和删除操作非常容易 (只需修改指针指向,时间复杂度为 O(1)O(1)O(1))。
📌 二、 C/C++ 动态内存分配底层对比
在实现线性表(尤其是顺序表的动态扩容)时,C 和 C++ 采用了不同的底层内存管理机制。
1. 核心语法对比
-
C 语言风格 :
cint *a = (int *)malloc(100 * sizeof(int)); // 显式计算字节数并进行强制类型转换 -
C++ 风格 :
cppint *a = new int[100]; // 语义更清晰,自动触发构造函数
2. 🔥 【红字防坑、核心方法总结】malloc(10) 的底层真相
手写笔记中记录了一个非常经典的易错点:
c
int *b = (int *)malloc(10); // 误区:申请 10 个 int 的空间?
❌ 常见误区 :初学者容易望文生义,认为括号里的 10 代表申请 10 个单位的空间(即 10 个 int)。
验证并补充如下底层逻辑:
-
malloc(参数)括号中的参数单位是 字节(Byte),而不是元素个数! -
在 32 位或 64 位的主流系统中,一个
int占用 4 字节 。因此,malloc(10)实际上仅仅申请了 10 个字节 的空间。 -
10 字节只能完整容纳 10÷4=210 \div 4 = 210÷4=2 个
int变量,还剩下 2 字节碎片。 -
正确申请 10 个
int空间的写法必须如第一句所示,配合sizeof运算符:cint *b = (int *)malloc(10 * sizeof(int)); // 实际申请 40 字节
3. 指针与引用标识符
*------ 标识符/间接访问符 :在指针变量声明时作为标识(如int *p),在表达式中作为解引用(如*p = 10)。&------ 取地址符:用于获取变量在内存中的起始物理地址。
📌 三、 算法复杂度深度评析(分值占比:45分,含13分代码题)
408 数据结构部分总分 45 分,其中固定包含一道 13 分的算法大题 。搞定这道题的关键在于平衡时间 与空间。
1. 时间复杂度 T(n)T(n)T(n)
- 本质 :算法执行所需的时间。
- 计算依据 :基本语句的频度之和 f(n)f(n)f(n)(即代码核心循环执行的次数)。
- 度量方法 :取数量级,使用大 OOO 表示法(如 O(1),O(n),O(n2)O(1), O(n), O(n^2)O(1),O(n),O(n2)),只保留最高阶项。
2. 空间复杂度 S(n)S(n)S(n)
- 本质 :算法执行所需的辅助空间。
- 🔥 【高频考点方法论】关于 O(1) 的极端要求 :
- 当题目规定"要求算法的时间复杂度优秀,而空间复杂度为 O(1)O(1)O(1) "时,这意味着所需辅助空间是一个常数。
- 无论问题规模 nnn 有多大,你只能定义常数个互不影响的指针或辅助变量(如
int i, temp;)。 - 严禁 在算法内部使用
malloc或new动态开辟一个与 nnn 相关的辅助数组,否则空间复杂度会飙升至 O(n)O(n)O(n),导致 13 分的代码题被重扣物理空间分。
考研复习是一场苦修,但脚踏实地走过的每一步,都是你最坚固的护城河。
如果今天的复盘对你有启发,欢迎 一键三连(点赞、收藏、关注) 支持一下!