王道数据结构课后代码题 p149 第3—— 7(c语言代码实现)

目录

3.编写后序遍历二叉树的非递归算法

 4.试给出二叉树的自下而上、自右到左的层次遍历算法 (有图解代码详解)c语言代码实现

5.假设二叉树采用二叉链表存储结构,设计一个非递归算法求二叉树的高度。

​编辑

 6.设一棵二叉树中各结点的值互不相同,其先序遍历序列和中序遍历序列分别存于两个一维数组A[l...n]和 B[l...n]中,试编写算法建立该二叉树的二叉链表。

 7.二叉树按二叉链表形式存储,写一个判别给定二叉树是否是完全二叉树的算法


3.编写后序遍历二叉树的非递归算法

 本题代码如下

void postorder(tree* t)
{struct treenode* stack[100];//初始化结构体数组int top = -1;//让栈顶指向-1treenode* p = *t;while (p || top != -1)//p不为空,并且栈不为空{if (p){top++;//p不为空,将p压入栈中stack[top] = p;p = p->lchild;//一直向左下遍历}else{p = stack[top];//p等于栈顶元素if (p->rchild && p->rchild->tag == 0)//右孩子不为空且未被访问过p = p->rchild;else//否则弹出结点并访问{p = stack[top];top--;printf("%c", p->data);p->tag = 1;//标记p被访问过p = NULL;}}}
}

完整测试代码

#include<stdio.h>
#include<stdlib.h>
typedef struct treenode
{char data;struct treenode* lchild, * rchild;int tag;
}treenode,*tree;
void buildtree(tree *t)
{char ch;ch = getchar();if (ch == '#')*t = NULL;else{*t = (treenode*)malloc(sizeof(treenode));(*t)->data = ch;(*t)->tag = 0;(*t)->lchild = NULL;(*t)->rchild = NULL;buildtree(&(*t)->lchild);buildtree(&(*t)->rchild);}
}
void postorder(tree* t)
{struct treenode* stack[100];//初始化结构体数组int top = -1;//让栈顶指向-1treenode* p = *t;while (p || top != -1)//p不为空,并且栈不为空{if (p){top++;//p不为空,将p压入栈中stack[top] = p;p = p->lchild;//一直向左下遍历}else{p = stack[top];//p等于栈顶元素if (p->rchild && p->rchild->tag == 0)//右孩子不为空且未被访问过p = p->rchild;else//否则弹出结点并访问{p = stack[top];top--;printf("%c", p->data);p->tag = 1;//标记p被访问过p = NULL;}}}
}
int main()
{tree t;buildtree(&t);postorder(&t);return 0;
}

用ABD##E##CF##G##

/*        A
    B        C
D      E   F     G    
*/

 4.试给出二叉树的自下而上、自右到左的层次遍历算法 (有图解代码详解)c语言代码实现

 本题我们采用让结点出队时将结点入栈,同时访问该结点,是否有左右孩子,如果有的话,就让左右孩子进队。最后所有结点都入栈了,再从栈顶开始依次访问就可以得到结果

看下面的图解

A先入队,然后出队,就压入栈中

访问A结点,有左右孩子,左右孩子入队

B结点出队并入栈,并访问B结点,B结点有左右孩子,左右孩子进队

C结点出队并入栈,同时访问C结点,C结点有左右孩子,左右孩子进队

D结点出队并入栈,同时访问D结点,D结点没有左右孩子

EFG依次出队进栈(与D的步骤相同)

最后我们看一下栈中的元素

我们让栈中元素依次出栈就能得到我们想要的结果

下面我们来看一下代码该如何实现:

void level(tree* t)
{treenode* q[10];treenode* s[10];int top = -1;int f = -1;int r = -1;treenode* p;q[++r] = *t;//根结点进队while (f < r){p = q[++f];//结点出队s[++top] = p;//结点进栈if (p->lchild)//出队结点是否有左孩子q[++r] = p->lchild;//有左孩子,左孩子进栈if (p->rchild)//出队结点是否有右孩子q[++r] = p->rchild;//有右孩子,右孩子进栈}while (top != -1)//依次输出栈中元素{printf("%c ", s[top--]->data);}
}

完整测试代码如下

#include<stdio.h>
#include<stdlib.h>
typedef struct treenode
{char data;struct treenode* lchild, * rchild;
}treenode,*tree;
void buildtree(tree* t)
{char ch;ch = getchar();if (ch == '#')*t = NULL;else{*t = (treenode*)malloc(sizeof(treenode));(*t)->data = ch;(*t)->lchild = NULL;(*t)->rchild = NULL;buildtree(&(*t)->lchild);buildtree(&(*t)->rchild);}
}
void level(tree* t)
{treenode* q[10];treenode* s[10];int top = -1;int f = -1;int r = -1;treenode* p;q[++r] = *t;while (f < r){p = q[++f];s[++top] = p;if (p->lchild)q[++r] = p->lchild;if (p->rchild)q[++r] = p->rchild;}while (top != -1){printf("%c ", s[top--]->data);}
}
int main()
{tree t;buildtree(&t);level(&t);return 0;
}

用ABD##E##CF##G##测试

5.假设二叉树采用二叉链表存储结构,设计一个非递归算法求二叉树的高度。

 采用层次遍历的算法,设置变量 ans记录当前结点所在的层数,设置变量 l 指向当前层的最右结点,每次层次遍历出队时与 l指针比较,若两者相等,则层数加 1,并让 l指向下一层的最右结点,直到遍历完成。ans的值即为二又树的高度。

本题代码如下

int deep(tree* t)//求树的深度
{if ((*t) == NULL)return 0; treenode* q[10];int f = -1, r = -1;//f头结点,r尾结点int l = 0, ans = 0;//l每次指向每层的最后一个结点q[++r] = *t;treenode* p;while (f < r){p = q[++f];//队列元素出队,正在访问的结点if (p->lchild)q[++r] = p->lchild;//左孩子入队if (p->rchild)q[++r] = p->rchild;//右孩子入队if (f == l)//处理该层的最右结点{ans++;//层数+1l = r;//让l指向下一层}}return ans;
}

完整测试代码

#include<stdio.h>  
#include<stdlib.h>  
typedef struct treenode
{char data;struct treenode* lchild, * rchild;
}treenode, * tree;
void buildtree(tree* t)//建树
{char ch;ch = getchar();if (ch == '#')*t = NULL;else{*t = (treenode*)malloc(sizeof(treenode));(*t)->data = ch;(*t)->lchild = NULL;(*t)->rchild = NULL;buildtree(&(*t)->lchild);buildtree(&(*t)->rchild);}
}
int deep(tree* t)//求树的深度
{if ((*t) == NULL)return 0; treenode* q[10];int f = -1, r = -1;//f头结点,r尾结点int l = 0, ans = 0;//l每次指向每层的最后一个结点q[++r] = *t;treenode* p;while (f < r){p = q[++f];//队列元素出队,正在访问的结点if (p->lchild)q[++r] = p->lchild;//左孩子入队if (p->rchild)q[++r] = p->rchild;//右孩子入队if (f == l)//处理该层的最右结点{ans++;//层数+1l = r;//让l指向下一层}}return ans;
}
int main()
{tree t;buildtree(&t);printf("树的高度为:%d\n", deep(&t)); return 0;
}
/*			AB        CD     E  F    */
//ABD##E##CF###

 6.设一棵二叉树中各结点的值互不相同,其先序遍历序列和中序遍历序列分别存于两个一维数组A[l...n]和 B[l...n]中,试编写算法建立该二叉树的二叉链表。

 本题代码如下

tree build(char a[], char b[], int s, int e)
{if (s <= e){treenode* root = (treenode*)malloc(sizeof(treenode));root->data = a[pos];//将子树的根节点赋值给rootint i;for (i = s; i <= e; i++)//在b数组中找到根节点if (b[i] == root->data)break;pos++;root->lchild = build(a, b, s, i - 1);//建立左子树root->rchild = build(a, b, i + 1, e);//建立右子树return root;}return NULL;
}

完整测试代码

#include<stdio.h>
typedef struct treenode {char data;struct treenode* lchild, * rchild;
}treenode,*tree;
int pos = 0;//全局变量pos
tree build(char a[], char b[], int s, int e)
{if (s <= e){treenode* root = (treenode*)malloc(sizeof(treenode));root->data = a[pos];//将子树的根节点赋值给rootint i;for (i = s; i <= e; i++)//在b数组中找到根节点if (b[i] == root->data)break;pos++;root->lchild = build(a, b, s, i - 1);//建立左子树root->rchild = build(a, b, i + 1, e);//建立右子树return root;}return NULL;
}
void disp(tree t)
{if (t){disp(t->lchild);disp(t->rchild);printf("%c", t->data);}
}
int main(){char a[] = {'A','B','D','E','C','F'};//先序char b[] = {'D','B','E','A','F','C' };//中序tree root = build(a, b, 0, 5);printf("后序序列为:");disp(root);return 0;
}

 7.二叉树按二叉链表形式存储,写一个判别给定二叉树是否是完全二叉树的算法

 采用层次遍历算法,将所有结点加入队列(包括空结点)。

如果没有左孩子,就看有没有右孩子,如果有右孩子,那么不为完全二叉树。

如果有左孩子,且之前不存在缺孩子的结点,左孩子进队,如果有右孩子,右孩子也进队,否则就是缺孩子了。之前存在缺孩子的,那么就不是完全二叉树。

有两种代码的写法

本题代码如下

int isok(tree* t)//判断完全二叉树
{squene q;q.f = q.r = q.tag = 0;int flag = false; // 标志是否遇到了空节点if (*t == NULL)return true; // 空树也是完全二叉树enquene(&q, *t);treenode* p;while (!isempty(&q)){dequene(&q, &p);if (p->lchild){if (flag) // 如果之前遇到了空节点,说明不是完全二叉树return false;enquene(&q, p->lchild);}else{flag = true;}if (p->rchild){if (flag) // 如果之前遇到了空节点,说明不是完全二叉树return false;enquene(&q, p->rchild);}else{flag = true;}}return true;
}

完整测试代码

#include <stdio.h>
#include <stdlib.h>
#define Max 15
#define true 1
#define false 0
typedef struct treenode
{char data;struct treenode* lchild, * rchild;
} treenode, * tree;
void buildtree(tree* t)
{char ch;ch = getchar();if (ch == '#')*t = NULL;else{*t = (treenode*)malloc(sizeof(treenode));(*t)->data = ch;buildtree(&(*t)->lchild);buildtree(&(*t)->rchild);}
}
typedef struct squene
{struct treenode* data[Max];int f, r, tag;
} squene;
int isempty(squene* q)//判断队空
{if (q->f == q->r && q->tag == 0)return true;return false;
}
int isfull(squene* q)//判断队满
{if (q->f == q->r && q->tag == 1)return true;return false;
}
int enquene(squene* q, treenode* p)//进队操作
{if (isfull(q))return false;q->data[q->r] = p;q->r = (q->r + 1) % Max;q->tag = 1;return true;
}
int dequene(squene* q, treenode** p)//出队操作
{if (isempty(q))return false;*p = q->data[q->f];q->f = (q->f + 1) % Max;q->tag = 0;return true;
}
int isok(tree* t)//判断完全二叉树
{squene q;q.f = q.r = q.tag = 0;int flag = false; // 标志是否遇到了空节点if (*t == NULL)return true; // 空树也是完全二叉树enquene(&q, *t);treenode* p;while (!isempty(&q)){dequene(&q, &p);if (p->lchild){if (flag) // 如果之前遇到了空节点,说明不是完全二叉树return false;enquene(&q, p->lchild);}else{flag = true;}if (p->rchild){if (flag) // 如果之前遇到了空节点,说明不是完全二叉树return false;enquene(&q, p->rchild);}else{flag = true;}}return true;
}
int main()
{treenode* t;buildtree(&t);if (isok(&t))printf("yes");elseprintf("no");return 0;
}

用ABD##E##CF##G##测试

/*                A

        B                C

D        E        F        G      

*/

用ABD###CF##G##测试

/*                A

        B                C

D                  F        G

*/

还可以用另外一种写法

#include <stdio.h>
#include <stdlib.h>
#define Max 15
typedef struct treenode
{char data;struct treenode* lchild, * rchild;
} treenode, * tree;
void buildtree(tree* t)
{char ch;ch = getchar();if (ch == '#')*t = NULL;else{*t = (treenode*)malloc(sizeof(treenode));(*t)->data = ch;(*t)->lchild = NULL;(*t)->rchild = NULL;buildtree(&(*t)->lchild);buildtree(&(*t)->rchild);}
}int isok(tree* t)//判断完全二叉树
{treenode* q[Max];int f = -1, r = -1;int tag = 1;//标记是否为完全二叉树q[++r] = *t;treenode* p;int flag = 1;//标记缺孩子if (*t == NULL) {tag = 1;}if (!(*t)->lchild && !(*t)->rchild)tag = 1;while (f < r) {p = q[++f];if (!p->lchild) //没有左孩子缺孩子{flag = 0;if (p->rchild)tag = 0;}else//有左孩子{if (flag)//之前不存在缺孩子的结点{q[++r] = p->lchild;if (p->rchild)q[++r] = p->rchild;elseflag = 0;}else//之前存在缺孩子的结点tag = 0;}}if (tag)return 1;return 0;
}
int main()
{treenode* t;buildtree(&t);if (isok(&t))printf("yes");elseprintf("no");return 0;
}

用ABD##E##CF##G##

/*                A

        B                C

D        E        F        G      

*/

测试结果为

用AB#E##CF###

/*                A

        B                C

            E        F   

*/

测试结果为

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

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

相关文章

普冉(PUYA)单片机开发笔记(7): ADC-轮询式多路采样

概述 应用中经常会有使用单片机进行模数转换的需求。PY32F003 具有 1 个 12 位的模拟数字转换器&#xff08;ADC&#xff09;&#xff0c;今天我们一起来使用一下这个 ADC。 数据手册中对 ADC 简介如下。 SAR ADC&#xff1a;逐次逼近式 ADC&#xff0c;原理参见“参考链接&a…

class070 子数组最大累加和问题与扩展-上【算法】

class070 子数组最大累加和问题与扩展-上【算法】 code1 53. 最大子数组和 // 累加和最大子数组和 // 给你一个整数数组 nums // 请你找出一个具有最大累加和的非空子数组 // 返回其最大累加和 // 测试链接 : https://leetcode.cn/problems/maximum-subarray/ dp[i]&#xff…

【Docker】Docker Compose,yml 配置指令参考的详细讲解

作者简介&#xff1a; 辭七七&#xff0c;目前大二&#xff0c;正在学习C/C&#xff0c;Java&#xff0c;Python等 作者主页&#xff1a; 七七的个人主页 文章收录专栏&#xff1a; 七七的闲谈 欢迎大家点赞 &#x1f44d; 收藏 ⭐ 加关注哦&#xff01;&#x1f496;&#x1f…

基于c++版数据结构基于数组栈改-Python思维总结

##栈部分-&#xff08;叠猫猫&#xff09; ##抽象数据类型栈的定义&#xff1a;是一种遵循先入后出的逻辑的线性数据结构。 换种方式去理解这种数据结构如果我们在一摞盘子中取到下面的盘子&#xff0c;我们首先要把最上面的盘子依次拿走&#xff0c;才可以继续拿下面的盘子&…

【Java期末复习资料】(2)常见例题 //持续更新

本文章主要是常见例题&#xff0c;解析不会太详细&#xff0c;有问题、不会的可以给我发消息哦&#xff0c;后续会出模拟卷 常见例题&#xff1a; 1.下列跟Java技术平台有关的是&#xff08;ABD&#xff09; A.JVM B.JDK C.JPN D.JRE 2.面向对象的特征包括&#xff08;ACD&…

wxPython的控件tree

wxPython树控件介绍 树&#xff08;tree&#xff09;是一种通过层次结构展示信息的控件&#xff0c;如下图所示是树控件示例&#xff0c;左窗口中是树控件&#xff0c;在wxPython中树控件类是wx.TreeCtrl。 wx.TreeCtrl常用的方法有 AddRoot(text, image-1, selImage-1, data…

在Deepin中安装x11vnc工具并结合内网穿透软件实现远程访问桌面

文章目录 1. 安装x11vnc2. 本地远程连接测试3. Deepin安装Cpolar4. 配置公网远程地址5. 公网远程连接Deepin桌面6. 固定连接公网地址7. 固定公网地址连接测试 x11vnc是一种在Linux系统中实现远程桌面控制的工具&#xff0c;它的原理是通过X Window系统的协议来实现远程桌面的展…

P4 Qt如何添加qss样式表文件和添加图片资源

目录 前言 01 添加图片资源文件 02 添加qss文件 前言 &#x1f3ac; 个人主页&#xff1a;ChenPi &#x1f43b;推荐专栏1: 《C_ChenPi的博客-CSDN博客》✨✨✨ &#x1f525; 推荐专栏2: 《Qt基础_ChenPi的博客-CSDN博客》✨✨✨ &#x1f33a;本篇简介 &#xff1a;这一章…

JVM Optimization Learning(六)

目录 一、JVM Optimization 1、Shenandoah Shenandoah的使用方法 2、ZGC ZGC的版本更迭 ZGC的使用方法 ZGC的参数设置 3、JMH测试GC性能 一、JVM Optimization 1、Shenandoah Shenandoah是由Red Hat开发的一款低延迟的垃圾收集器&#xff0c;Shenandoah并发执行大部分…

机器人纯阻抗控制接触刚性环境(阻尼影响因素)

问题描述 在机器人学中&#xff0c;阻抗控制是一种常用的控制策略&#xff0c;用于管理机器人在与环境交互时的运动和力。阻抗控制背后的关键概念是将环境视为导纳&#xff0c;而将机器人视为阻抗。 纯阻抗控制接触刚性环境时&#xff0c;机器人的行为方式主要受其阻抗参数的…

【开源】基于Vue和SpringBoot的车险自助理赔系统

项目编号&#xff1a; S 018 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S018&#xff0c;文末获取源码。} 项目编号&#xff1a;S018&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 角色管理模块2.3 车…

Maven基础

目录 Maven坐标 坐标简介 主要组成 Maven依赖管理 配置依赖 依赖简介 配置依赖 依赖传递 依赖传递简介 排除依赖 依赖范围 生命周期 生命周期简介 执行指定生命周期 Maven坐标 坐标简介 Maven中的坐标是资源的唯一标识&#xff0c;通过该坐标可以唯一定位资…

Python开发运维:Python调用K8S API实现资源管理

目录 一、实验 1.Python操作K8S API获取资源 2.Python操作K8S API创建deployment资源 3.Python操作K8S API删除k8s资源 4.Python操作K8S API修改k8s资源 5.Python操作K8S API查看k8s资源 二、问题 1.Windows11安装kubernetes报错 2.Python通过调用哪些方法实现Pod和De…

在SpringData JPA 中实现对持久层的操作

1.导入依赖 hibernate 这个依赖自带实现JPA接口 <dependencies><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope></dependency><depen…

CrystalDiskInfo中文版(硬盘检测工具) v9.1.1.0 绿色汉化版-供大家学习研究参考

更新内容 重新支持三星SATA SSD寿命报告 增加对ZHITAI SC001的支持 新增SK hynix Gold S31支持 增加了KLEVV NEO N610的支持。 改进的Micron/Crucial SATA SSD支持 已更改 卸载程序将显示一个确认对话框&#xff0c;用于删除设置。 强大功能 1.拥有多国语言&#xff0c;…

045:Vue读取本地上传JSON文件,导出JSON文件方法

第045个 查看专栏目录: VUE ------ element UI 专栏目标 在vue和element UI联合技术栈的操控下&#xff0c;本专栏提供行之有效的源代码示例和信息点介绍&#xff0c;做到灵活运用。 &#xff08;1&#xff09;提供vue2的一些基本操作&#xff1a;安装、引用&#xff0c;模板使…

jquery手写广告轮播图,无限循环功能

说明 在很多情况下&#xff0c;我们都需要开发广告轮播图&#xff0c;当我们进行页面的功能开发时&#xff0c;采用轮播图来实现也行&#xff0c;但是很多情况下&#xff0c;我们只需要简单的控制轮播循环轮播展示即可&#xff0c;所以用jq开开发广告轮播波&#xff0c;自定义…

Idea 插件开发: Swing Designer设计器创建的组件全部为空问题记录

问题现象 通过Swing 设计器创建的对象, Swing组件全部是空的, 导致ToolWindowFactory工厂的实现类调用时候出现了空指针异常 如下方式创建的 问题分析 问题出现时候, 同时给我生成了一个createUIComponents的私有方法, 由于个人当时理解有误, 把他当成了初始化方法, 在里面…

Oracle高可用一家老小全在这里

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…

用Java实现一对一聊天

目录 服务端 客户端 服务端 package 一对一用户; import java.awt.BorderLayout; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.io.PrintWriter; import java.net.ServerSocket; import java.net.Socket; imp…