《数据结构知识点总结.docx》由会员分享,可在线阅读,更多相关《数据结构知识点总结.docx(32页珍藏版)》请在三一办公上搜索。
1、1数据结构概述数据结构是计算机科学中的一门基础课程,它研究数据组织和处理的方式。数据结构是一种逻辑上的组织形式,它描述了数据元素之间的关系,并定义了在这些数据元素上执行的操作。数据结构可以分为线性结构和非线性结构两种:1线性结构:数据元素之间的关系是一对一的关系,例如数组、链表、栈、队列等。2非线性结构:数据元素之间的关系是一对多的关系,例如树、图等。数据结构的设计和实现关系到程序的性能和可维护性,常用的数据结构包括:1数组:一组相同数据类型的数据按照一定的顺序排列,通过下标来访问。2链表:数据元素之间通过指针相连,可以是单向链表、双向链表或循环链表。3栈:一种先进后出的数据结构,只允许在栈顶
2、进行插入和删除操作。4队列:一种先进先出的数据结构,只允许在队尾插入元素,在队头删除元素。5树:由节点和边组成的数据结构,每个节点有零个或多个子节点,一般用于模拟具有层次关系的问题。6图:由节点和边组成的数据结构,每个节点有零个或多个相邻节点,用于模拟复杂的网络结构。2数据结构的相关概念、名词和术语在学习数据结构时,需要了解一些相关的概念、名词和术语,以下是一些常见的:1数据元素:组成数据结构的基本单位。2数据项:数据元素中的单个数据,例如一个人的身高、体重等。3结点:树和图中的基本单位,包含数据元素和指向其他结点的指针。4根结点:树中没有父结点的结点,是树的起点。5叶子结点:树中没有子结点的
3、结点,是树的终点。6父结点:有子结点的结点。7子结点:有父结点的结点。8树的度:树中一个结点所拥有的子树的个数。9树的深度:从根结点到最深叶子结点的路径长度。10链表:由结点按线性顺序连接而成的数据结构。11单向链表:每个结点只有一个指向下一个结点的指针。12双向链表:每个结点同时具有指向上一个结点和下一个结点的指针。13循环链表:尾结点的指针指向头结点,形成一个循环。14栈:一种先进后出的数据结构。15队列:一种先进先出的数据结构。16堆:一种可以快速找到最大或最小元素的树形数据结构。17散列表:一种通过将关键字映射到表中一个位置来访问记录的数据结构。18排序算法:对数据元素进行排序的算法,
4、例如冒泡排序、快速排序、归并排序等。3数据的逻辑结构、存储结构。在数据结构中,数据的逻辑结构和存储结构是两个基本概念。1.数据的逻辑结构:数据的逻辑结构指的是数据元素之间的逻辑关系,它决定了数据元素之间的组织方式和操作方式。常见的数据逻辑结构包括线性结构、树形结构和图形结构。线性结构包括线性表、栈和队列,其中线性表是一种有序的数据元素序列,栈是一种限定插入和删除操作只能在同一端进行的线性表,队列是一种插入操作只能在一端进行,删除操作只能在另一端进行的线性表。树形结构包括二叉树、多叉树和森林,其中二叉树是每个结点最多只有两个子节点的树形结构,多叉树是每个结点有多个子节点的树形结构,森林是多个互不
5、相交的树的集合。图形结构是由结点和边组成的非线性结构,结点表示数据元素,边表示数据元素之间的关系。图形结构包括有向图和无向图,其中有向图的边具有方向性,无向图的边没有方向性。1.数据的存储结构:数据的存储结构指的是数据元素在计算机内存中的存储方式,包括顺序存储和链式存储两种。顺序存储是指将数据元素存储在一段连续的存储空间中,数据元素之间的物理地址是连续的。顺序存储适用于线性表和一些简单的树形结构,如满二叉树。链式存储是指将数据元素存储在任意的存储空间中,通过指针来链接相邻的数据元素,数据元素之间的物理地址是不连续的。链式存储适用于线性表、树形结构和图形结构。4算法的概念,算法的效率评价。1.算
6、法的概念:算法是一组定义良好的指令,用于解决特定问题或执行特定任务的有限步骤序列。简单来说,算法就是解决问题的方法和步骤。在计算机科学中,算法是计算机程序的核心部分,它决定了程序的执行效率和正确性。因此,设计高效的算法对于计算机科学的发展至关重要。1.算法的效率评价:算法的效率评价是指对算法的执行效率进行度量和评估,常用的评价方法包括时间复杂度和空间复杂度。时间复杂度是指算法执行所需要的时间,通常用算法中基本操作的执行次数来度量。时间复杂度用大。记号表示,比如0(n)、O(nlogn),0(M2)等。其中,n表示输入数据的规模,时间复杂度越小,算法执行效率越高。空间复杂度是指算法在执行过程中所
7、需要的存储空间,通常用算法中数据存储量的大小来度量。空间复杂度也用大0记号表示,例如0(1)、0(n)、0(M2)等。其中,n表示输入数据的规模,空间复杂度越小,算法所需的存储空间越少。5线性表线性表是最基本、最常用的数据结构之一,它是n(n=0)个具有相同特性的数据元素的有限序列。在线性表中,数据元素的逻辑关系是一对一的关系,即除了第一个和最后一个元素,其余元素都有唯一的直接前驱和直接后继。线性表通常用1.来表示,其中1.O表示第一个元素,1.I表示第二个元素,以此类推,1.n-I表示第n个元素,n为线性表的长度。线性表的实现方式有两种,分别是顺序存储和链式存储。顺序存储是将线性表的元素顺序
8、地存放在一块连续的存储空间中,即用一组地址连续的存储单元依次存储线性表的元素。由于数据元素在存储空间中的位置是连续的,因此可以随机访问线性表中的任意元素。链式存储是将线性表的元素存放在任意的存储单元中,通过指针来链接相邻的元素,即每个元素都有一个指针指向下一个元素。由于数据元素在存储空间中的位置是不连续的,因此访问线性表中的元素需要通过指针进行遍历。线性表支持的基本操作包括:1初始化:初始化线性表,即创建一个空的线性表。2插入:将新的元素插入到线性表的指定位置上。3删除:删除线性表中指定位置的元素。4查找:查找线性表中指定位置的元素。5修改:修改线性表中指定位置的元素。6遍历:按照线性表元素的
9、顺序遍历整个线性表。6线性表的顺序存储结构和链式存储结构及其基本操作的实现。线性表是一种基本的数据结构,它包括顺序存储结构和链式存储结构两种形式。其中,顺序存储结构使用一段连续的存储空间来存储线性表中的元素,而链式存储结构则使用一组节点来存储线性表中的元素,每个节点包括数据域和指针域。1.顺序存储结构的基本操作(1)初始化线性表:即为线性表中的元素分配一段连续的存储空间,并将线性表的长度设置为0。(2)插入元素:在指定位置插入一个新元素,需要将该位置之后的元素向后移动一个位置,并修改线性表的长度。(3)删除元素:删除指定位置的元素,需要将该位置之后的元素向前移动一个位置,并修改线性表的长度。(
10、4)查找元素:在线性表中查找指定元素的位置,需要遍历整个线性表进行搜索。(5)修改元素:修改指定位置的元素。1.链式存储结构的基本操作(1)初始化线性表:即为线性表中的元素分配一组节点,并将头指针指向第一个节点。(2)插入元素:在指定位置插入一个新元素,需要先找到该位置的前驱节点,然后将新节点插入到其后面,并修改前驱节点的指针域和新节点的指针域。(3)删除元素:删除指定位置的元素,需要先找到该位置的前驱节点,然后将其指针域指向该位置的后继节点,并释放该位置的节点。(4)查找元素:在线性表中查找指定元素的位置,需要遍历整个链表进行搜索。(5)修改元素:修改指定位置的元素。7栈,栈的顺序存储和链式
11、存储及其基本运算的实现栈是一种常见的数据结构,它具有后进先出(1.lFO)的特点,即最后入栈的元素最先出栈。栈的基本运算包括入栈和出栈,其中入栈将一个元素放到栈顶,而出栈则将栈顶元素取出。1.栈的顺序存储栈的顺序存储可以使用数组实现。定义一个数组作为栈的存储空间,再定义一个指针top,指向栈顶元素的位置。初始时,top指向-1,表示栈为空。(1)初始化栈:将top置为-1,表示栈为空。(2)入栈操作:将元素压入栈顶,即将top加1,然后将元素存储在top所指向的位置。(3)出栈操作:将栈顶元素取出,即先将top指向的元素取出,然后将top减Io(4)获取栈顶元素:返回top指向的元素。1.栈的
12、链式存储栈的链式存储可以使用单向链表实现。定义一个节点表示栈的元素,包含一个数据域和一个指针域,指向下一个节点。再定义一个指针top,指向栈顶元素所在的节点。初始时,top指向空节点,表示栈为空。(I)初始化栈:将top置为空节点,表示栈为空。(2)入栈操作:新建一个节点,将元素存储在数据域中,然后将该节点的指针域指向top所指向的节点,最后将top指向新节点。(3)出栈操作:将top指向的节点取出,然后将top指向该节点的下一个节点。(4)获取栈顶元素:返回top指向的节点的数据域。8队列。队列的顺序存储(循环队列)和链式存储及其基本运算的实现。队列是一种常见的数据结构,它具有先进先出(Fl
13、Fo)的特点,即最先进入队列的元素最先出队列。队列的基本运算包括入队和出队,其中入队将一个元素加入队尾,而出队则将队头元素取出。1.队列的顺序存储(循环队列)队列的顺序存储可以使用数组实现,常见的是循环队列。定义一个数组作为队列的存储空间,再定义两个指针front和rear,分别指向队列的队头和队尾。初始时,front和rear均指向0,表示队列为空。(1)初始化队列:将front和rear均置为0,表示队列为空。(2)入队操作:将元素加入队尾,即将元素存储在rear所指向的位置,然后将rear加1。如果rear已经到达数组的末尾,还需要将rear置为0,实现循环队列的效果。(3)出队操作:将
14、队头元素取出,即先将front指向的元素取出,然后将front加1。如果front已经到达数组的末尾,还需要将front置为0,实现循环队列的效果。(4)获取队头元素:返回front指向的元素。1.队列的链式存储队列的链式存储可以使用单向链表实现。定义一个节点表示队列的元素,包含一个数据域和一个指针域,指向下一个节点。再定义两个指针front和rear,分别指向队列的队头和队尾所在的节点。初始时,front和rear均指向空节点,表示队列为空。(1)初始化队列:将front和rear均置为空节点,表示队列为空。(2)入队操作:新建一个节点,将元素存储在数据域中,然后将该节点的指针域指向空节点,
15、再将rear指向该节点。如果队列为空,还需要将front指向该节点。(3)出队操作:将front指向的节点取出,然后将front指向该节点的下一个节点。如果队列为空,返回错误信息。(4)获取队头元素:返叵front指向的节点的数据域。9串。串的基本运算的实现。串(String)是由零个或多个字符组成的有限序列,常见的表示方法是用单引号或双引号括起来的字符序列。在计算机中,串是一种基本数据类型,也是计算机程序中常用的数据类型之一。串的基本运算包括串的连接、求子串、串的比较、查找子串等。1.串的连接串的连接就是将两个串拼接起来,得到一个新的串。在程序中,可以使用循环或指针来实现串的连接。例如:c+
16、QCopycodecharstl=hello”;charstr2=world*;charstr312;inti=0,j=0:while(strliI=0:)str3j+=strli+;)i=0;while(str2i!=0)str3j+=str2i+;)s3j=0;字符串末尾需要添加0表示结束1.求子串子串是原串中任意个连续字符组成的串。求子串的过程可以使用指针来实现,例如:charstr=llo,world*:charsub6;intpos=2:/从第3个字符开始取5个字符intIen=5:char*p=str+pos;/P指向Str中的第3个字符for(inti=0:iIen;i+)sub
17、i=*(p+i);/取5个字符存入SUb中SUbnen=,0,:/字符串末尾需要添加0表示结束1.串的比较串的比较是指比较两个串的大小关系。可以使用循环或指针来实现串的比较,例如:c+QCopycodecharstrl=hell匚.:charstr2=world*;inti=0:while(strli=str2i&strli!=,0,str2iI=0)i+;if(stxlistr2i)/strl大于Str2else/Strl等于Str21.查找子串查找子串是指在一个字符串中查找另一个字符串是否出现,并返回其位置。可以使用循环或指针来实现子串的查找,例如:charstr=hello,world;
18、charsub=wor;int,jzk;for(i=0;stri!=0;i+)k=O;while(strj!=O,&subk!=,O,&strj=subk)j+;k+;)if(subk=,O)/子串匹配成功返回子串的起始位置在上述代码中,变量i表示在主串中开始查找的位置,j表示当前匹配的位置,k表示当前匹配的字符在子串中的位置。在循环中,当主串和子串的当前字符相等时,j和k同时后移;当子串匹配完全时,说明匹配成功,返回子串在主串中的起始位置。除了以上基本运算外,还可以实现串的替换、插入、删除等操作。10树和森林在数据结构中,树和森林是两个常用的概念。树是一种非常常见的数据结构,它由一个根节点和
19、零个或多个子树组成。每个子树也是一个树,它们之间通过连接父节点和子节点的边进行连接。树通常用于表示具有层次结构的数据,例如目录结构、组织结构等。森林是由多个树组成的集合。森林中的每个树都是独立的,它们之间没有任何连接。森林通常用于表示多个独立的数据结构,例如多个无关联的家谱、多个无关联的组织结构等。在树和森林中,节点通常包括一个数据元素和一个指向其子节点的指针。根节点是整个树的起点,没有父节点。叶子节点是没有子节点的节点。路径是从根节点到任何一个节点的边的集合。深度是指从根节点到某个节点的路径长度。子树是由某个节点及其所有后代组成的树。常见的树包括二叉树、二叉搜索树、AV1.树、红黑树等。常见
20、的森林包括哈夫曼树、并查集等。11二叉树及完全二叉树的基本性质二叉树是一种特殊的树形数据结构,它每个节点最多只有两个子节点,分别称为左子节点和右子节点。以下是二叉树的基本性质:1 .第i层最多有2A(i-l)个节点。2 .深度为k的二叉树最多有2”-1个节点。3 .对于任意一棵二叉树,如果其叶子节点数为n,度为2的节点数为2,则0=n2+lo4 .在任意一棵非空二叉树中,若n为节点总数,n为叶子节点数,nl为度为1的节点数,则n=n+nl+l15 .一棵深度为k,且有2”-1个节点的二叉树,称为满二叉树。6 .一棵深度为k,且有n个节点的二叉树,当且仅当其每一个节点都与深度为k的满二叉树中,编
21、号为1至n的节点一一对应时,称之为完全二叉树。完全二叉树有着一些特殊的性质,包括:1.对于任意的i(l=il,则其父节点是节点i/2(整除)。 如果2in,则节点i没有左子节点。 如果2i+ln,则节点i没有右子节点。1.对于一棵含有n个节点的完全二叉树,它的高度为Iog2(n).12二叉树的存储结构二叉树的存储结构主要有两种:顺序存储和链式存储。1.顺序存储结构顺序存储结构是将二叉树的节点按照某种顺序存储在一维数组中。一般按照从上到下、从左到右的顺序进行存储,根节点存储在数组下标为1的位置,其左子节点存储在下标为2的位置,右子节点存储在下标为3的位置,以此类推。如果某个节点没有左子节点或右子
22、节点,则其对应的数组元素值为null或0。由于顺序存储结构需要连续的内存空间,因此对于非满二叉树,会造成内存浪费。1.链式存储结构链式存储结构是将二叉树的节点存储在堆中,并通过指针进行连接。每个节点都包含三个属性:数据域、左子节点指针、右子节点指针。根据左子节点指针和右子节点指针的指向,可以实现对二叉树的遍历和操作。链式存储结构不需要连续的内存空间,因此能够更好地利用内存,适用于非满二叉树。13二叉树的遍历二叉树的遍历是指按照某种规则依次访问二叉树中的每个节点,遍历的方式分为三种:前序遍历、中序遍历和后序遍历。以下是它们的详细解释和示例代码:1.前序遍历前序遍历的访问顺序是:先访问根节点,然后
23、依次遍历左子树和右子树。具体步骤如下:(1)访问当前节点。(2)递归遍历当前节点的左子树。(3)递归遍历当前节点的右子树。代码实现:python白Copycodedefpreorder_traversal(root):ifnotroot:returnprint(root.val)#访问当前节点preorder-traversal(root,left)#递归遍历左子树preorder-traversal(root,right)#递归遍历右子树1.中序遍历中序遍历的访问顺序是:先遍历左子树,然后访问根节点,最后遍历右子树。具体步骤如下:(1)递归遍历当前节点的左子树。(2)访问当前节点。(3)递归
24、遍历当前节点的右子树。代码实现:definorder-traversal(root):ifnotroot:returninorder-traversal(root,left)。递归遍历左子树print(root.val)#访问当前节点inorder-traversal(root,right)#递归遍历右子树1.后序遍历后序遍历的访问顺序是:先遍历左子树,然后遍历右子树,最后访问根节点。具体步骤如下:(1)递归遍历当前节点的左子树。(2)递归遍历当前节点的右子树。(3)访问当前节点。代码实现:python白Copycodedefpostorde-riveral(root):ifnotroot:r
25、eturnpostorder_traversal(root,left)#递归遍历左子树posterder_traversal(root,right)tt递归遍历右子树print(root.val)#访问当前节点以上三种遍历方式是二叉树的基本遍历方式,它们都是递归实现的,时间复杂度为0(n),其中n是节点的数量。14树转换为二叉树将一棵树转换为二叉树是指将原树的节点重新排列,形成一棵二叉树。具体实现方式是,对于原树中的每个节点,将其所有子节点都看作是它的右孩子,将该节点的兄弟节点看作是它的左孩子,然后再将该节点与其左孩子之间添加一个空节点,最终形成的二叉树就是所求。下面给出一个示例来说明这个过程
26、。假设原树如下所示:CSS其中,将原树中的B节点变成了A的左孩子,C节点变成了A的右孩子,D节点变成了B的左孩子,E节点变成了B的右孩子,F节点变成了C的右孩子。在B和C之间添加了一个空节点。需要注意的是,在对节点进行遍历的过程中,需要记录当前节点的前驱节点P,以便将P与当前节点的左孩子连接起来,同时也需要记录根节点,以便返回最终的二叉树。下面是Python代码的实现:classTreeNode:def_init_(self,val=0,Ieft=Nonezright=None):self.val=valself.left=leftself.right=rightdeftree_to_bina
27、ry(root):ifnotroot:returnNonenode=TreeNode(root.val)node.left=Nonenode.right=tree_to_binary(root.left)ifnode.rightisNone:node.right=tree_to_binary(root.right)else:p=node.rightwhilep.rightisnotNone:P=p.rightp.right=tree_to_binary(root,right)returnnode#示例root=TreeNodefA)root,left=TreeNodefB)root.right
28、=TreeNode(lC)root.left,left=TreeNode(,D,)root,right.left=TreeNode(E,)root.right.right=TreeNode(F,)binary_root=tree_to_binary(root)最终得到的二叉树为:15构建哈夫曼树及设计哈夫曼编码哈夫曼树是一种用于编码压缩的树形数据结构。该树结构的构建是通过对输入数据进行频率分析来实现的,频率越高的字符在哈夫曼树中越靠近根节点。根据哈夫曼树的特性,每个叶子节点都对应着一个输入字符,并且该叶子节点到根节点的路径表示该字符的哈夫曼编码。哈夫曼树的构建可以通过以下步骤来实现:1 .统计
29、输入数据中每个字符出现的频率,并将它们存储在一个表格中;2 .将频率表格中的每个字符都视为一个独立的节点,并将它们存储在一个未组成树的集合中;3 .从节点集合中找到两个频率最低的节点,将它们合并为一个新的节点,并将该节点的频率设置为两个原节点的频率之和;4 .将新的节点添加到节点集合中,同时从集合中删除原节点;5 .重复步骤3和步骤4,直到集合中只剩下一个节点,即为哈夫曼树的根节点。构建好的哈夫曼树可以用于生成哈夫曼编码,方法如下:1 .从哈夫曼树的根节点开始,依次遍历每个节点;2 .如果遍历到的节点是一个叶子节点,则记录下该节点的哈夫曼编码;3 .如果遍历到的节点是一个内部节点,则不记录任何
30、编码,继续向下遍历;4 .重复步骤1到步骤3,直到遍历完整个哈夫曼树。生成的哈夫曼编码是一个二进制字符串,由。和1组成,每个字符对应一个唯一的编码。由于哈夫曼树的构建方式保证了频率高的字符的编码比频率低的字符的编码短,因此使用哈夫曼编码进行压缩可以有效地减少数据的存储空间。16树和森林的存储表示、转化方法与遍历。树和森林是常用的数据结构,它们有不同的存储表示、转化方法和遍历方式。下面分别介绍一下它们的相关知识:1树的存储表示树可以通过链式存储或者数组存储来表示。链式存储方式是指每个节点都有一个指向其子节点的指针或者指向其父节点的指针。这种方式比较灵活,但是需要动态分配内存,容易造成空间浪费。数
31、组存储方式是指将树中的所有节点按照某种顺序存储在一个一维数组中。一般采用层次遍历的方式,将每层节点依次存储在数组中,这样可以用一段连续的内存空间存储整棵树,可以节省空间。但是,如果树中存在空节点,会造成数组中的空间浪费。2树的转化方法有时候需要将一种树结构转化为另一种树结构。常用的树结构转化方法包括:(1)左孩子右兄弟表示法左孩子右兄弟表示法(也称为二叉树表示法“)是一种将一般树转化为二叉树的方法。该方法将树中的每个节点视为二叉树的节点,将每个节点的第一个孩子视为其左孩子,将其兄弟视为其右孩子。这样,就可以将一般树转化为二叉树,便于进行遍历等操作。森林转化为二叉树森林是多棵树的集合,可以将其转
32、化为一棵二叉树。具体方法是将每棵树的根节点作为二叉树节点的左孩子,将其右边的树作为其右孩子。3树的遍历树的遍历方式主要有前序遍历、中序遍历和后序遍历。(1)前序遍历前序遍历是指先访问根节点,再访问左子树,最后访问右子树。前序遍历的递归实现可以如下所示:scssQCopycodevoidPreorderTraversaKTreetree)if(tree)(printf(*%dtreedata);PreorderTraversal(tree-left):PreordexTraversal(tree-right):(2)中序遍历中序遍历是指先访问左子树,再访问根节点,最后访问右子树。中序遍历的递归实
33、现可以如下所示:scssQCopycodevoidInorderTraversal(Treetree)(if(tree)InorderTraversal(tree-left);printf(%d”,tree-data);InorderTraversal(tree-right);后序遍历后序遍历是指先访问左子树,再访问右子树,最后访问根节点。后序遍历的递归实现可以如下所示:scssQCopycodevoidPostorderTraversal(Treetree)if(tree)PostorderTraversal(tree-left):PostorderTraversal(tree-right)
34、;printf(%d”,tree-data):在遍历过程中,可以使用栈或队列等数据结构来存储需要遍历的节点。前序遍历和中序遍历可以使用栈来实现,后序遍历可以使用两个栈来实现。4森林的存储表示森林可以通过链式存储或者数组存储来表示。链式存储方式与树的链式存储方式类似,每个节点都有一个指向其子节点的指针或者指向其兄弟节点的指针。数组存储方式可以采用类似于二叉树表示法的方式,将每棵树的根节点存储在数组中,将其左孩子存储在其下一个位置,将其右兄弟存储在数组中的下一个位置。5森林的遍历森林的遍历方式和树的遍历方式类似,只是需要对每棵树都进行遍历。可以使用递归或者栈等数据结构来实现。其中,先序遍历和后序遍
35、历可以使用递归实现,中序遍历可以使用栈实现。17图的存储表示(邻接矩阵、邻接表),图的遍历方法图的存储表示:1.邻接矩阵:用一个二维数组来表示,其中数组的每个元素都表示一个节点之间的连接情况,如果节点i和节点j之间有边,则相应的数组元素值为1,否则为Oo2.邻接表:用一个数组和一些链表来表示,数组中的每个元素表示一个节点,链表中的每个节点表示与该节点相连的其他节点,链表中存储的是该节点的邻居节点的信息,例如相邻节点的编号和边的权值。图的遍历方法:1.深度优先遍历(DFS):从图的某个顶点开始遍历,访问该顶点,并依次访问该顶点的所有未被访问过的邻居节点,直到所有邻居节点都被访问为止,然后返回上一
36、个访问的节点,继续进行同样的遍历操作,直到图中的所有节点都被访问过为止。2.广度优先遍历(BFS):从图的某个顶点开始遍历,首先访问该顶点,并将该顶点的所有邻居节点加入一个队列中,然后依次访问队列中的所有节点,并将这些节点的未被访问过的邻居节点加入队列中,直到队列为空为止,此时图的所有节点都被访问过了。18如何用prim算法和kruskal算法构建图的最小生成树Prim算法和Kruskal算法都是用于构建图的最小生成树的算法。Prim算法的步骤如下:1首先选择一个起始节点,将其加入到最小生成树中,并将该节点的所有相邻边加入到一个优先队列中(优先队列中的边按照权值从小到大排列)。2从优先队列中取
37、出权值最小的边,如果该边所连接的节点不在最小生成树中,则将该节点加入到最小生成树中,并将该节点的所有相邻边加入到优先队列中。3重复执行第二步,直到最小生成树包含了所有的节点。Kruskal算法的步骤如下:1将所有的边按照权值从小到大排序。2依次取出排序后的边,如果该边所连接的两个节点不在同一个连通分量中,则将该边加入到最小生成树中,并将这两个节点所在的连通分量合并。3重复执行第二步,直到最小生成树包含了所有的节点。19查找在数据结构中,查找是指在一个数据集合中寻找一个特定的元素或满足一定条件的元素的过程。常用的查找算法有线性查找、二分查找、哈希查找等。1线性查找:线性查找也称为顺序查找,是一种
38、最基本的查找算法。它的思路是从数据集合的第一个元素开始,逐个比较元素的值,直到找到目标元素或遍历完整个数据集合。2二分查找:二分查找也称为折半查找,是一种效率较高的查找算法。它的前提是数据集合中的元素必须是有序的,因此需要先对数据进行排序。然后在查找过程中,每次将待查找区间折半,判断目标元素在哪一半,从而缩小查找范围,直到找到目标元素或确认目标元素不存在。3哈希查找:哈希查找也称为散列表查找,是一种利用哈希函数将目标元素映射到一个固定的地址上的查找算法。在哈希查找过程中,将目标元素的值作为输入,通过哈希函数计算出对应的地址,然后在该地址上查找目标元素。除了上述算法外,还有一些特殊的查找算法,如
39、:树查找、广义表查找、快速查找、插值查找等。20顺序查找顺序查找也称为线性查找,是一种最基本的查找算法。它的思路是从数据集合的第一个元素开始,逐个比较元素的值,直到找到目标元素或遍历完整个数据集合。顺序查找的具体步骤如下:1从数据集合的第一个元素开始,逐个比较元素的值。2如果找到目标元素,则返回该元素在数据集合中的位置。3如果遍历完整个数据集合仍未找到目标元素,则返回查找失败的信息。顺序查找算法的时间复杂度为O(n),其中n为数据集合中元素的个数。由于顺序查找不需要对数据进行排序,因此适用于不经常变动的数据集合或者数据集合较小的情况。顺序查找的优点是简单易实现,缺点是效率较低,不适用于数据集合
40、较大的情况。21折半查找和分块查找的算法折半查找(BinarySearCh)是一种在有序数组中查找指定元素的算法,它的思想是将数组分成两半,然后判断目标元素应该在哪一半中,然后继续将这一半继续折半查找,直到找到目标元素或者发现数组中已经没有目标元素为止。以下是使用C语言实现的折半查找算法:cQCopycodeintbinaxySearch(in:arr,intlow,inthigh,inttarget)while(lowidtarget)low=mid+1:elsehigh=*id-1;)return-1;/没有找到目标元素,返回1.分块查找(BIoCkSearch)是一种在分块有序数组中查找
41、指定元素的算法,它的思想是将数组分成若干个块,每个块中的元素有序,然后通过二分查找定位到目标元素所在的块,然后在该块中进行顺序查找,直到找到目标元素或者遍历完所有块为止。以下是使用C语言实现的分块查找算法:defineB1.OCK,SIZE4/定义若个块中元素的个数intblocksearch(intarr,intblockArr,intn,inttarget)intblockNum=n/B1.OCK-SIZE+(n%B1.OCK.SIZE=070:1):/计算数组被分成的块白intblockindex=binarySearch(blockArr,0,blockNum-1,target):/二
42、分查找目标元素Iif(blockindex=1)return-1;/没有找到目标元素,返回intlow=blockindexB1.OCK_SIZE;inthigh=low+B1.OCK,SIZE-1:for(inti=low:i=highAitarget):/二分查找目标元素所在的块:blockindex=-1)return-1;/没有找到目标元素,返回-1low=blockindex*B1.OCK_SIZE;high=low+B1.OCK_SIZE-1:(inti=low:i=highMtikey=key;root-left=root-right=NU1.1.;elseif(keykey)/插入到左子树root-left=insertBST(root-left,key):elseif(keyroot-key)/插入到右子树root-right=insertBST(root-right,key);returnroot;节点的删除操作:1如果待删除的节点是叶子节点,直接删除即可;2如果待删除的节点只有一个子节点,将它的子节点替代它的位置;3如果待删除的节点有两个子节点,找到它的右子树的最小节点或左子树的最大节点,将它的关键字替代待删除节点的关键字,并删除该最小节点或最大节点。以下是使用C语言实现的二叉排序树的节点的删除操作:BSTNo