树的BFS:广度优先搜索,层次遍历的实现步骤

BFS是树的经典遍历方法,按“广度优先”(层次)访问节点,核心依赖队列(FIFO)实现。其步骤为:初始化队列(根节点入队),循环取出队首节点访问,将左、右子节点(或子节点自然顺序)入队,直至队空。 BFS适用于树的层次关系问题,如计算树高、判断完全二叉树、求根到叶最短路径等。以二叉树 `1(2(4,5),3)` 为例,层次遍历顺序为1→2→3→4→5。 关键点:队列确保层次顺序,子节点入队顺序(左→右),时间复杂度O(n)(n为节点数),空间复杂度O(n)(最坏队列存n/2节点)。掌握BFS可高效解决层次问题,为复杂算法奠基。

阅读全文
树的DFS:深度优先搜索,从根到叶的遍历方法

树由节点和边组成,每个节点(除根外)有一个父节点,可有多子节点。DFS(深度优先搜索)是“深入一条路到黑再回溯”的遍历方法。树的DFS遍历含前序(根→左→右)、中序、后序,前序最直接体现根到叶路径。 递归实现前序遍历:访问当前节点→递归左子树→递归右子树。以示例树(根1,左2、右3;2左4、右5)为例,顺序为1→2→4→5→3。非递归用栈:初始化栈入根,循环弹出栈顶访问,先右后左入栈子节点。 根到叶DFS遍历用于路径求和、输出路径等问题。递归实现直观,非递归用栈模拟适合大数据。掌握前序遍历是树结构核心技能。

阅读全文
哈希函数:哈希函数如何生成哈希值?初学者必知

哈希函数是将任意长度输入转为固定长度哈希值的“翻译器”,哈希值即数据的“身份证号”。其核心特点:固定长度(如MD5为32位十六进制字符)、单向不可逆(无法由哈希值反推原数据)、近似唯一(碰撞概率极低)、雪崩效应(输入微小变化致哈希值巨变)。生成过程分三步:输入预处理为二进制,分段数学运算,合并结果。与加密函数不同,哈希单向无需密钥,加密可逆需密钥。应用广泛:文件校验(比对哈希值防篡改)、密码存储(存哈希值保安全)、数据索引及分布式系统数据分布。哈希如数据指纹,关键特性使其在安全与校验中不可或缺。

阅读全文
插入排序:插入排序如何工作?与冒泡排序对比
2025-12-20 11 阅读 数据结构

文章介绍了排序的基础重要性,并聚焦两种简单排序算法:插入排序和冒泡排序。 插入排序核心思想是逐步构建有序序列,从第二个元素开始,将每个元素插入到已排序部分的正确位置(类似整理扑克牌)。其平均时间复杂度O(n²),最好情况(数组有序时)为O(n),空间复杂度O(1),稳定,适用于小规模或接近有序数据。 冒泡排序则通过相邻元素比较,将大元素逐步“冒”到末尾(如气泡上浮),每轮确定一个最大元素的位置。平均时间复杂度同样O(n²),空间复杂度O(1),稳定,但移动元素更多,实际应用较少。 两者均为O(n²)复杂度,插入排序更高效,尤其数据接近有序时表现更好。理解它们是学习复杂排序算法的基础。

阅读全文
二分查找:二分查找的适用场景,零基础也能学会

这篇文章介绍了二分查找算法,其核心是在有序数组中通过比较中间元素,逐步缩小查找范围,快速定位目标。它适用于有序、大数据量、静态(少修改)且需快速查找的场景,如字典或配置文件。 查找过程通过左右指针`left`和`right`确定中间值`mid`,根据目标与中间值的大小调整指针:若中间值等于目标则找到;若目标更大,右移`left`;若更小,左移`right`,直至找到或范围无效。 Python迭代实现的核心代码通过`left <= right`循环,计算`mid = (left + right)//2`,边界处理确保数组为空或目标不存在时返回-1。时间复杂度为O(log n)(每次范围减半),空间复杂度为O(1)(仅用常数变量)。 关键细节包括处理重复元素需扩展遍历,单元素数组直接判断,找不到目标返回-1。二分查找的“减治”思想高效解决有序大数据的快速查找问题,是算法基础中的重要工具。

阅读全文
邻接矩阵:图的另一种表示方法,优缺点对比

邻接矩阵是图的一种基础表示方式,本质为n×n二维数组,行与列对应图的顶点,元素值表示顶点间边的存在性或权重。无向图中,元素1表示有边,0表示无边;有权图则直接存储权重值。 其优点:一是判断边存在仅需O(1)时间,计算顶点度高效(无向图行和,有向图行/列分别对应出/入度);二是适合稠密图(边数接近n²),空间利用率高,且实现简单,便于初学者理解。 缺点:空间复杂度为O(n²),稀疏图时浪费大量空间;遍历邻接顶点需O(n)时间,效率低于邻接表;动态调整边数灵活性不足。 总结:邻接矩阵以空间换时间,适合稠密图、需频繁查询边或计算度的场景,不适合稀疏图或需频繁遍历邻接顶点的场景,是理解图结构的基础工具。

阅读全文
并查集:并查集是什么?解决“朋友关系”问题的方法

并查集是管理元素分组的高效数据结构,核心解决“合并组”(Union)和“查询是否同组”(Find)问题,适用于快速判断元素是否属于同一集合的场景。其底层以parent数组维护父节点关系,每组视为一棵树,根节点为组代表,初始各元素自成一组。 关键优化是**路径压缩**(查询时压缩路径,使节点直接指向根)和**按秩合并**(小树依附大树,避免树退化为链表),确保操作接近常数时间复杂度。核心方法`find`(查找根节点并压缩路径)和`union`(合并两组,小树根指向大树根)实现高效分组。 应用广泛,如网络连接判断、家族关系查询、最小生成树(Kruskal算法)及等价类问题等,是处理分组场景的简洁强大工具。

阅读全文
前缀和:前缀和数组如何快速计算区间和?

前缀和数组是一种用于快速计算区间和的辅助数组。定义为:原数组A的前缀和数组S,其中S[0]=0,S[k](k≥1)是A[1]到A[k]的和,即S[k] = S[k-1] + A[k]。例如原数组A=[1,2,3,4,5],其前缀和数组S=[0,1,3,6,10,15]。 计算区间和的核心公式是:原数组从第i个到第j个元素的和 = S[j] - S[i-1]。例如计算A[2]到A[4]的和,用S[4]-S[1]=10-1=9,结果正确。 优势在于:预处理S数组需O(n)时间,每次区间和查询仅需O(1)时间,总复杂度O(n+q)(q为查询次数),远快于直接计算的O(qn)。需注意索引对齐(如原数组从0开始时公式调整)、区间合法性及空间换时间的特点。 前缀和数组通过“提前累积”实现

阅读全文
动态规划:动态规划入门,斐波那契数列的高效解法

斐波那契数列定义为f(0)=0,f(1)=1,n>1时f(n)=f(n-1)+f(n-2)。直接递归计算时,因重复计算过多,时间复杂度达O(2^n),效率极低。 动态规划通过空间换时间优化:一是记忆化递归,用备忘录数组存储已计算结果,每个子问题仅算一次,时间与空间复杂度均为O(n);二是递推法,仅用两个变量迭代计算,时间O(n)、空间O(1),为最优解。 动态规划核心特性是重叠子问题(子问题重复出现)与最优子结构(当前解依赖子问题解)。其本质是通过存储子问题结果避免重复计算,斐波那契是经典入门案例,掌握后可推广至爬楼梯等同类问题。

阅读全文
平衡二叉树:为什么需要平衡?旋转操作简单讲

二叉搜索树(BST)因极端插入可能退化为链表,操作复杂度升至O(n)。平衡二叉树通过**平衡因子**(节点左右子树高度差)控制平衡,要求平衡因子为-1、0或1。当不平衡时,通过**旋转操作**(LL右旋、RR左旋、LR先左旋后右旋、RL先右旋后左旋)调整结构,使树高保持log n级别,确保查找、插入、删除等操作复杂度稳定在O(log n)。旋转本质是调整支点,恢复树的平衡结构。

阅读全文
图:图的基本概念,邻接表表示法初学者指南

图由顶点(点)和边(连接关系)组成,顶点是基本单元,边可单向(有向图)或双向(无向图),有权图边带权重(如距离),无权图仅存连接关系。邻接表是高效表示法,解决邻接矩阵在稀疏图(边远少于顶点平方)中空间浪费问题,核心是每个顶点对应存储直接相连顶点的列表。无向图邻接表如顶点0连接1、2、3,邻接表为[1,2,3];有权图可存“邻接点+权重”元组。邻接表空间复杂度O(V+E)(V顶点数,E边数),适合稀疏图,遍历邻接点方便,但判断两点是否有边需遍历邻接表。掌握邻接表为图遍历、最短路径等算法奠定基础。

阅读全文
堆:堆的结构与应用,最小堆和最大堆入门

堆是一种特殊的完全二叉树,核心特点是父节点与子节点满足大小关系(最小堆父≤子,最大堆父≥子),能高效获取极值(堆顶为最小或最大元素),类似优先队列。其底层为完全二叉树,每一层尽量填满,最后一层从左到右排列。数组存储时,左子节点索引=2i+1,右子节点=2i+2,父节点=(i-1)//2。基本操作包括插入(末尾添加后上浮)和删除(堆顶删除后尾元素顶替,再下沉),时间复杂度均为O(log n)。堆广泛用于优先队列(任务调度)、找第k大元素、哈夫曼编码等场景,是高效处理极值问题的关键结构。

阅读全文
贪心算法:贪心算法是什么?找零钱问题实例讲解

贪心算法是每步选择当前最优(局部最优)以期望全局最优的算法,核心是满足“贪心选择性质”——每步局部最优能导致全局最优。经典应用如找零钱:以25、10、5、1分硬币为例,找67分时,按从大到小逐步取25×2(50分)、10×1(10分)、5×1(5分)、1×2(2分),共6枚,验证为最优。其局限性在于,若问题不满足贪心选择性质(如硬币面额[1,3,4]找6分),贪心可能失效(贪心取4+1+1=3枚,最优为3+3=2枚)。适用场景包括硬币面额合理(如25、10、5、1)、活动安排(选最早结束活动)等。总之,贪心算法简单直观高效,但仅适用于满足贪心选择性质的问题,不保证所有问题全局最优。

阅读全文
分治算法:分治思想如何解决问题?归并排序原理

分治算法核心是“分而治之”,通过分解(拆分为小问题)、解决(递归求解子问题)、合并(整合结果)三步处理复杂问题,适合递归结构场景。以数组总和计算为例,分解数组,递归求子数组和,合并得总和。 归并排序是典型应用:先分解数组至单个元素(本身有序),再用双指针法合并有序子数组。其时间复杂度O(n log n),空间复杂度O(n)(需临时数组)。 分治通过递归简化问题,归并排序高效体现其优势,是理解递归、排序等算法的基础。

阅读全文
递归:递归是什么?用斐波那契数列举例,初学者也能懂

这篇文章用生活化的例子和经典案例讲解了递归的概念。递归是将大问题分解为更小的同类问题,直到问题小到可直接解决(终止条件),再通过小问题结果反推大问题结果的方法,核心是“分解”与“终止”。 以斐波那契数列为例,其递归定义为:F(0)=0,F(1)=1,n>1时F(n)=F(n-1)+F(n-2)。计算F(5)时,需先算F(4)和F(3),依此类推,直到分解到F(0)或F(1)(终止条件),再逐层返回结果。递归的关键是必须有明确终止条件(如n=0、1),且每次调用规模递减,否则会无限循环。 Python代码实现简洁:`def fibonacci(n): if n==0: return 0; elif n==1: return 1; else: return fibonacci(n-1)+fibonacci(n-2)`。递归代码优雅但计算大数字(如F(100))时效率低于迭代法,体现了“以退为

阅读全文
查找算法:顺序查找和二分查找的区别,哪个更快?

文章介绍了两种基础查找算法:顺序查找和二分查找,用于解决从数据中定位特定元素的问题。 顺序查找(线性查找)原理是逐个比较元素,无需数据有序,时间复杂度O(n)(n为数据量),优点是简单,缺点是效率低,适合小数据量或无序数据。 二分查找(折半查找)要求数据有序,通过分半比较缩小范围,时间复杂度O(log n),效率高(如n=1000时仅需约10次),但需处理边界条件,适合大数据量有序数据。 两者对比:顺序查找无需有序、实现简单但效率低;二分查找需有序且复杂度高但速度快。选择依据为数据规模和有序性:有序大数据用二分,无序小数据用顺序。

阅读全文
排序算法:冒泡排序入门,步骤详解+代码示例

冒泡排序是计算机科学中最简单的排序算法之一,核心思想是通过重复比较相邻元素并交换位置,使大元素逐步“冒泡”到数组末尾。其基本步骤为:外层循环控制n-1轮比较(每轮确定一个大元素位置),内层循环从第一个元素开始,依次比较相邻元素,若前大后小则交换;优化项为若某轮无交换,说明数组已有序,可提前终止。 时间复杂度上,最坏情况(完全逆序)为O(n²),最好情况(已排序)为O(n),空间复杂度为O(1)(仅需常数额外空间)。该算法实现简单、易于理解,适合小规模数据排序,是排序算法的入门基础。

阅读全文
哈希表:哈希表如何存储数据?冲突解决方法图解

哈希表是通过哈希函数将键映射到数组桶位置的键值对存储结构,实现O(1)级别的高效查找、插入与删除。其底层为数组,键经哈希函数(如“键%数组长度”)转换为数组索引(桶位置),直接存储对应的值。 当不同键哈希值相同时会发生冲突(如学号12和22在数组长度10时均%10得2)。经典解决方法有二:链地址法(每个桶存链表,冲突元素挂在链表尾部),实现简单但需额外空间;开放定址法(线性探测下一个空桶,如冲突位置h→h+1→h+2...),纯数组操作但可能形成聚集。 哈希表核心在于哈希函数、冲突处理逻辑,是数据结构学习的基础。

阅读全文
二叉树:二叉树的三种遍历方式,递归实现超简单

这篇文章介绍了二叉树的三种经典遍历方式(前序、中序、后序),基于递归实现,核心是明确根节点的访问位置。 二叉树每个节点最多有左右子树,遍历即按特定顺序访问节点。递归是关键,类似“套娃”,函数自调用且范围缩小,直到遇到空节点终止。 三种遍历顺序区别:前序(根→左→右)、中序(左→根→右)、后序(左→右→根)。以示例树(根1,左2,右3;2的左4,右5)为例: - 前序遍历结果:1 2 4 5 3; - 中序遍历结果:4 2 5 1 3; - 后序遍历结果:4 5 2 3 1。 递归实现核心:终止条件(空节点返回)+ 按遍历顺序递归左右子树。通过明确根位置和递归逻辑,可清晰理解遍历过程。

阅读全文
树:树结构是什么?用生活例子轻松理解

这篇文章用生活类比讲解数据结构中的“树”。核心是树与生活中的树类似:有根节点(起点)、子节点/父节点(分支与源头)、叶子节点(无后代)及子树(节点与后代),具有非线性、分支型、层级分明的特点。 与线性链表(单一路径)不同,树可多分支(如根节点分多个子节点)。生活中树结构无处不在:家庭关系以长辈为根,公司架构以CEO为根,电脑文件系统以磁盘为根,均体现层级分支。 树的核心优势是高效处理层级化分支问题,如数据库索引、导航路径规划、游戏场景构建等。理解树结构能掌握分支型问题的处理思维,生活中家庭、公司、文件系统都是树的典型应用。

阅读全文
队列:队列的“先进先出”如何实现?简单例子说明

队列是遵循“先进先出”(FIFO)原则的数据结构,仅能在队尾入队、队头出队,核心概念包括队头(最早元素)、队尾(最晚元素),基本操作为入队(Enqueue)和出队(Dequeue)。 以数组实现为例,需front(队头指针)、rear(队尾指针)及固定容量数组。队空条件为front == rear,队满为rear == max_size;入队时rear后移存储元素,出队时front后移取出元素。 实例演示:容量5的队列,初始front=0、rear=0;入队1、2、3后rear=3,队列[1,2,3];出队1(front=1),再入队4(rear=4);入队5后队列满,出队2(front=2),最终队列[3,4,5]。 应用场景包括任务调度、广度优先搜索(BFS)、打印机队列、网络请求等,在数据处理和任务排队中作用关键。

阅读全文
栈:栈的“后进先出”是什么意思?原理图解

这篇文章以“叠盘子”为例,解释了数据结构“栈”的核心概念。栈是只能从一端(栈顶)进行插入和删除操作的线性表,另一端为栈底,其核心特性是“后进先出”(LIFO)——最后放入的元素最先被取出。 栈的基本操作包括:入栈(push,添加元素到栈顶)、出栈(pop,移除并返回栈顶元素)、查看栈顶(top)和判空(empty)。例如,叠盘子时,新盘子放在最上面(入栈),拿取时必须先取最上面的(出栈),符合LIFO。 栈在生活与编程中广泛应用:括号匹配(用栈记录左括号,遇右括号弹出匹配)、函数调用栈(后调用的函数先返回)、浏览器后退功能(依次弹出最近访问的网页)等。理解栈的“LIFO”特性,能帮助解决递归、动态规划等问题,是数据结构的基础工具。

阅读全文
链表:单链表与双链表的区别,初学者一看就会

文章以游戏玩家列表存储为例,说明链表解决数组删除中间元素需移动节点的问题。链表是由节点组成的线性结构,节点含数据域和指针域,非连续内存存储,插入删除仅需修改指针。 单链表最简单,节点仅含next指针,单向遍历(从头至尾),插入删除需先找前驱节点改指针,省内存,适合单向场景(如队列)。 双链表节点多一个prev指针,支持双向遍历,插入删除直接通过prev/next指针操作,无需找前驱,内存稍高,适合双向操作(如浏览器历史、通讯录)。 单双链表对比:单链表结构简单省内存,双链表功能全但稍占内存。根据需求选择:单向用单链表,双向或频繁操作用双链表。

阅读全文
数组:为什么数组是数据结构的基石?零基础必学

这篇文章介绍了数组作为数据结构基础的核心地位。数组是相同类型元素的序列,通过索引(从0开始)实现随机访问,具有简单直观、连续存储和高效索引访问的特点。它是栈、队列、哈希表等复杂结构的基础(如栈用数组实现后进先出,队列用循环数组实现先进先出),也是二维数组(矩阵)的基础。数组支持遍历、查找、排序等基础操作,且随机访问时间复杂度为O(1),远超链表的O(n)。但它存在固定大小(静态数组)和插入删除效率低(需移动元素)的局限。总之,数组是数据结构的“入门钥匙”,掌握它能为后续学习复杂结构和算法奠定基础。

阅读全文
C++静态成员:类的共享变量与函数

这篇文章介绍了C++中静态成员(变量和函数)的概念、用法及注意事项。 静态成员用于解决普通成员变量无法共享数据的问题:静态成员变量(`static`修饰)属于整个类,存储在全局数据区,所有对象共享,需在类外初始化(如`int Student::count = 0;`),可通过类名或对象访问(如`Student::count`)。示例中`Student`类用静态变量`studentCount`统计对象数量,构造时加1、析构时减1,展示共享特性。 静态成员函数同样用`static`修饰,属于类而非对象,无`this`指针,只能访问静态成员,可通过类名或对象调用(如`Student::getCount()`)。 注意事项:静态成员变量需类外初始化;静态函数不能直接访问非静态成员;避免过度使用静态成员以降低耦合。 总结:静态成员实现类共享数据与工具函数,提升数据一致性,适用于全局状态(如计数器),但需合理控制使用场景。

阅读全文