数据结构代码题------树相关day01 序幕

2024-02-09 18:10

本文主要是介绍数据结构代码题------树相关day01 序幕,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

数据结构代码题----树

题目01

已知一颗二叉树,按照顺序存储结构进行存储,设计一个算法,求编号分别i和j的两个节点的最近的公共祖先结点的值。

算法分析

1、首先是对二叉树的顺序结构进行介绍
在这里插入图片描述
结合以上的图片描述,对于二叉树中,顺序存储的设计性质如下:

  1. 祖先结点的下标为i/2或者j/2
  2. 已知一个祖先结点,其儿子结点的下标为i*2

2、寻找最近公共祖先结点

思路:
想象下,现在二叉树的深度非常的大,若i和j分别在不同的子树系统中,若要找到最近的公共祖先结点,要怎么做呢???

答案

答案每次进行i和j进行比较,选择较大者进行比较并取i/2或者j/2进行设计,进行递归,直到最后的i=j便是最近的最佳的公共直接点。

其实现代码如下:

typedef ElemType int; 
ElemType Search_Common_Ancester(int A[], int i,int j){if(A[i] != NULL && A[j] != NULL){//设置起始条件while(i != j){if(i > j){i = i/2;//进行寻找祖先}else{j = j/2;//进行交换}}//这里退出了循环,因此有i=jreturn A[i];//返回结果
}}

核心代码:

//设置起始条件while(i != j){if(i > j){i = i/2;//进行寻找祖先}else{j = j/2;//进行交换}}//这里退出了循环,因此有i=jreturn A[i];//返回结果

递归实现:

typedef ElemType int; 
ElemType Search_Common_Ancester(int A[], int i,int j){if(A[i] == NULL  || A[j] == NULL){return 0;}if(i == j){return A[i];//递归出口}if(i < j){//j的位置大,因此需要对其进行向上寻找Search_Common_Ancester(A,i,j/2);}else{Search_Common_Ancester(A,i/2,j);}
}

题目02

二叉树中序遍历-----非递归实现

分析
结合下面的例子图进行讲解:

在这里插入图片描述
中序遍历非递归的设计口诀:

入栈向左一直走,出栈访问右子树

算法思路:

  1. 初始化一个数据结构栈
  2. 对于从根节点开始,依次入栈左子树的左节点,直到结点为NULL,
  3. 同时对其进行栈顶元素的出栈
  4. 出栈之前要先判断右子树是否存在
  5. 右子树存在则入栈
  6. 不存在则出栈并访问。

其实现的流程图大致如下:

在这里插入图片描述
结合以上的分析对其进行给出核心代码如下:

while(p != NULL || !isEmpty(S)){if(p != NULL){//入栈向左一直走push(S,p);p = p->lchild;}else{//当到达空结点pop(S,p);visit(p->data);//标记访问结点p = p->rchild;//出栈访问右子树}

完整代码:

typedef struct TNode{ElemType data;struct TNode* lchild, *rchild;
}TNODE;
//中序遍历
void MiddleTraverse(Tree root){InitStack(S);root =  p;
//起始条件while(p != NULL || !isEmpty(S)){if(p != NULL){//入栈向左一直走push(S,p);p = p->lchild;}else{//当到达空结点pop(S,p);visit(p->data);p = p->rchild;}}
}

题目03
二叉树后序遍历----非递归!!!

分析

二叉树的后序遍历是三种遍历序列中非递归实现最为困难的一个,需要对出栈元素进行多次判断,这里以上面的二叉树的进行分析:
在这里插入图片描述
后序遍历的实现依旧是需要借助于数据结构栈进行操作,因此我们试着先将其左子树入栈,向左一直走如下图:
在这里插入图片描述
大致实现的思路便是:

  1. 入栈左子树直到左子树为NULL
  2. 获取到栈顶的结点p-----采用GetTop(S,p)
  3. 对栈顶的结点进行右子树是否访问以及是否为NULL的检测,
  4. !!!!!!!!!!记住!!!
  5. 这里是对栈顶结点的右子树的是否为空和是否之前被访问过进行判断。
  6. 若不为空也没有被访问到,则该元素进栈,指向其左子树进行判断
  7. 若为空或者已经被访问到了,则栈顶元素直接出栈,并设置标记已经访问过
  8. 循环这个过程。

助记口诀:入栈向左一直走,判定(右子树),出栈访问,标记,重置初始指针

下面是核心实现代码:

	//核心代码if(p!=NULL){//入栈向左一直走push(S,p);//入栈p = p->lchild;}else{//这时p为空,说明左子树走到尽头了GetTop(S,p);//获取栈顶元素//对栈顶元素进行检测if(p->rchild != NULL && p->rchild != r){//不为空,没有被访问p = p->rchild;//转到右子树push(S,p);//入栈p = p->lchild;//左子树继续}else{//若已经被访问pop(S,p);//出栈visit(p->data);//访问r = p;//标记已经访问p=NULL;//重置为NULL方便下一次使用}

完整代码:


typedef struct TNode{ElemType data;struct TNode* lchild, *rchild;
}TNODE,*BiTree;
//非递归方法实现后序遍历
void OrderTaverse(BiTree root){if(root == NULL){return ;}InitStack(S);//初始化一个栈TNode * p = root;//操作指针TNode * r = NULL;//标记访问指针while(! isEmpty(S)){//核心代码if(p!=NULL){//入栈向左一直走push(S,p);//入栈p = p->lchild;}else{//这时p为空,说明左子树走到尽头了GetTop(S,p);//获取栈顶元素//对栈顶元素进行检测if(p->rchild != NULL && p->rchild != r){//不为空,没有被访问p = p->rchild;//转到右子树push(S,p);//入栈p = p->lchild;//左子树继续}else{//若已经被访问pop(S,p);//出栈visit(p->data);//访问r = p;//标记已经访问p=NULL;//重置为NULL方便下一次使用}}}
}

题目04

二叉树层次遍历非递归实现

其遍历实现图如下:
在这里插入图片描述
结合其二叉树的递归实现顺序,我们发现,这个实现的遍历结果采用队列的方式,对于其先进先出的特性非常符合,因此层次遍历采用队列实现

其实现过程图如下:
在这里插入图片描述
实现思路:

  1. 根节点判断是否为NULL
  2. 根节点入队,
  3. 出队访问左右子树
  4. 若左子树不为空,则访问左子树
  5. 若右子树不为空,访问右子树
  6. 队列为空,则循环停止

实现核心代码如下:

	while(!isEmpty(Q)){DeQueue(Q,p);//出队列visit(p->data);//访问数据//有左子树入队列if(p->lchild != NULL){EnterQueue(Q,p->lchild);}//右子树入队列if(p->rchild != NULL){EnterQueue(Q,p->rchild);}}

完整代码:

typedef struct TNode{ElemType data;struct TNode* lchild, *rchild;
}TNODE,*BiTree;
//层次遍历代码
void BehindTraverse(BiTree root){if(root == NULL){return ;}InitQueue(Q);//初始化队列TNODE *p;//根节点如队列EnterQueue(Q,root);while(!isEmpty(Q)){DeQueue(Q,p);//出队列visit(p->data);//访问数据//有左子树入队列if(p->lchild != NULL){EnterQueue(Q,p->lchild);}//右子树入队列if(p->rchild != NULL){EnterQueue(Q,p->rchild);}}
}

…二叉树后续代码题持续编写更新,祝大家1024程序员节日快乐!!!!

注:

个人代码问题或需要程序编写辅导服务等问题请加闲鱼【代码无bug】
或点击下面链接跳转闲鱼进行咨询

闲鱼链接

在这里插入图片描述

这篇关于数据结构代码题------树相关day01 序幕的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/694939

相关文章

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Redis实现高效内存管理的示例代码

《Redis实现高效内存管理的示例代码》Redis内存管理是其核心功能之一,为了高效地利用内存,Redis采用了多种技术和策略,如优化的数据结构、内存分配策略、内存回收、数据压缩等,下面就来详细的介绍... 目录1. 内存分配策略jemalloc 的使用2. 数据压缩和编码ziplist示例代码3. 优化的

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

Python从Word文档中提取图片并生成PPT的操作代码

《Python从Word文档中提取图片并生成PPT的操作代码》在日常办公场景中,我们经常需要从Word文档中提取图片,并将这些图片整理到PowerPoint幻灯片中,手动完成这一任务既耗时又容易出错,... 目录引言背景与需求解决方案概述代码解析代码核心逻辑说明总结引言在日常办公场景中,我们经常需要从 W

使用Spring Cache本地缓存示例代码

《使用SpringCache本地缓存示例代码》缓存是提高应用程序性能的重要手段,通过将频繁访问的数据存储在内存中,可以减少数据库访问次数,从而加速数据读取,:本文主要介绍使用SpringCac... 目录一、Spring Cache简介核心特点:二、基础配置1. 添加依赖2. 启用缓存3. 缓存配置方案方案

MySQL的配置文件详解及实例代码

《MySQL的配置文件详解及实例代码》MySQL的配置文件是服务器运行的重要组成部分,用于设置服务器操作的各种参数,下面:本文主要介绍MySQL配置文件的相关资料,文中通过代码介绍的非常详细,需要... 目录前言一、配置文件结构1.[mysqld]2.[client]3.[mysql]4.[mysqldum

Python多线程实现大文件快速下载的代码实现

《Python多线程实现大文件快速下载的代码实现》在互联网时代,文件下载是日常操作之一,尤其是大文件,然而,网络条件不稳定或带宽有限时,下载速度会变得很慢,本文将介绍如何使用Python实现多线程下载... 目录引言一、多线程下载原理二、python实现多线程下载代码说明:三、实战案例四、注意事项五、总结引

redis数据结构之String详解

《redis数据结构之String详解》Redis以String为基础类型,因C字符串效率低、非二进制安全等问题,采用SDS动态字符串实现高效存储,通过RedisObject封装,支持多种编码方式(如... 目录一、为什么Redis选String作为基础类型?二、SDS底层数据结构三、RedisObject