[树] △ 由广义表构造树(孩子链表CTree)并以广义表的形式输出(严蔚敏《数据结构》6.75|6.76 )

本文主要是介绍[树] △ 由广义表构造树(孩子链表CTree)并以广义表的形式输出(严蔚敏《数据结构》6.75|6.76 ),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目来源:严蔚敏《数据结构》C语言版本习题册 6.75、6.76

【题目】6.75
试写以递归算法,由6.73题定义的广义表表示法的字符序列,构造树的孩子链表。

【题目】6.76
试写以递归算法,以6.73题给定的树的广义表表示法的字符序列形式输出以孩子链表表示的树。

【测试数据】A(B(E,F),C(G),D)
【答案】

/*-------------------------|6.75 用广义表构造树     |-------------------------*/
// @Quesion:有一些格式检测不了"A(" "A()" "A)("
Status CreateCTreeByGList(CTree *pT, int parent) {// 创建新结点newNode --> 放在下标为pT->n// 该结点的爸爸为parentchar c;CNode *p, *q;int newNode;//创建newNode结点newNode = pT->n; //新结点的下标for (c=getchar(); c!='\n'; c=getchar() ) {if (c>='A' && c<='Z') { // 结点信息pT->nodes[newNode].data = c; //给结点赋值pT->nodes[newNode].firstchild = NULL; //给结点赋值pT->n++; //结点数+1//newNode有爸爸,即parentif (parent!=-1) {//创建孩子结点p = (CNode *)malloc(sizeof(CNode));if (!p) exit(OVERFLOW);p->index = newNode;p->next = NULL;//儿子父亲相认if (pT->nodes[parent].firstchild==NULL) {pT->nodes[parent].firstchild = p;} else {for (q=pT->nodes[parent].firstchild; q->next; q=q->next) ;q->next = p;}}} else if (c=='(') { //是newNode的孩子CreateCTreeByGList(pT, newNode); //开始创建newNode的孩子} else if (c==',') { //是newNode的兄弟,即parent的下一个孩子CreateCTreeByGList(pT, parent); //parent的下一个孩子return OK; //newNode结点构造完成(自己创建了、孩子创建了、兄弟创建了)} else if (c==')') { //parent构造完毕return OK; } else {return ERROR; //格式错误}}return OK;
}/*-------------------------|6.76 以广义表的形式输出 |-------------------------*/
Status PrintAsGList(CTree T,int parent) {CNode *p;if (T.n<=0) return ERROR; visit(T.nodes[parent].data);if (T.nodes[parent].firstchild) {printf("(");for (p=T.nodes[parent].firstchild; p; p=p->next) {PrintAsGList(T, p->index);if (p->next) printf(",");}printf(")");}return OK;
}

【完整代码】

#include<stdio.h>
#include<stdlib.h>
#include<string.h>#ifndef BASE
#define BASE
#define TRUE 1
#define FALSE 0
#define OK 1
#define ERROR 0
#define INFEASIBLE -1
#define OVERFLOW -2
typedef int Status;
typedef int bool;
#endif#define TElemType char
void visit(TElemType e) {printf("%c", e);
}
#define MAX_TREE_SIZE 100
#define maxSize 50typedef struct CNode{int index; //这个孩子的结点号(注意:在严书中变量名为child)struct CNode *next; //下一个孩子结点
}CNode, *ChildPtr; //孩子结点结构(在严书中名为CTNode)
typedef struct{TElemType data;CNode* firstchild;
}PNode; //双亲结点结构(在严书中,结构名为CTBox)
typedef struct{PNode nodes[MAX_TREE_SIZE];int n,r; //结点数 和 根结点的位置
}CTree; //树结构// 先根遍历
void SubPreOrder(CTree T, int index) {CNode *child;visit(T.nodes[index].data);for (child=T.nodes[index].firstchild; child; child=child->next)SubPreOrder(T, child->index);
}
void PreOrder(CTree T) {SubPreOrder(T, T.r);
}/*-------------------------|6.63 求树的深度         |-------------------------*/
int SubTreeDepth(CTree T, int index) { //序号为index的子树深度int max=-1; //孩子的最大深度int sd; //孩子的深度CNode *p;if (!T.nodes[index].firstchild) return 1; //没有孩子,深度为1for (p=T.nodes[index].firstchild; p; p=p->next) { //遍历该结点的所有孩子sd = SubTreeDepth(T, p->index); //求孩子的深度if (max<sd) max=sd;}return max+1; //孩子的最大深度+1
}
int TreeDepth(CTree T) { return SubTreeDepth(T, T.r);
}/*-------------------------|6.72 将树打印成树状     |-------------------------*/
void PrintAsTree(CTree T, int index, int i) {/*思路1. 观察题目输出的序列ABEFCGD2. 此为树的先根遍历–>对应为二叉树存储的先序遍历3. 前面的空格是该结点所在的层数*/CNode *p;int cnt;//输出空格for (cnt=1; cnt<i; cnt++) printf(" ");//输出元素visit(T.nodes[index].data);printf("\n");//遍历它的孩子for(p=T.nodes[index].firstchild; p; p=p->next)PrintAsTree(T, p->index, i+1);
}// 树的层序次序+每个结点的度 --> 创建CTree
Status CreateCTreeByLevelDegree(CTree *pT,char *levelstr, int *degree) {CNode *c,*sibling;int parent;int i,cnt;//创建结点for (i=0; i<strlen(levelstr); ++i) {//赋值pT->nodes[i].data = levelstr[i];pT->nodes[i].firstchild = NULL;}pT->n=strlen(levelstr); //个数pT->r=0; //根结点//为孩子找爸爸parent=0; //当前的爸爸i=1; //遍历孩子cnt=0; //已经为parent找到了cnt个孩子while (i<strlen(levelstr)) {if (degree[parent]==0 || cnt==degree[parent]) { //parent没有孩子 || parent的孩子已经全部找到cnt=0;parent++;continue;}cnt++; //为parent找到了一个孩子//创建孩子结点c = (CNode *)malloc(sizeof(CNode)); if (!c) exit(OVERFLOW);c->index = i; //孩子的编号c->next = NULL;if (cnt==1) { //第一个孩子pT->nodes[parent].firstchild = c;} else { //不是第一个孩子for(sibling=pT->nodes[parent].firstchild; sibling->next; sibling=sibling->next) ;sibling->next = c;}i++;}return TRUE;
}/*-------------------------|6.75 用广义表构造树     |-------------------------*/
// @Quesion:有一些格式检测不了"A(" "A()" "A)("
Status CreateCTreeByGList(CTree *pT, int parent) {// 创建新结点newNode --> 放在下标为pT->n// 该结点的爸爸为parentchar c;CNode *p, *q;int newNode;//创建newNode结点newNode = pT->n; //新结点的下标for (c=getchar(); c!='\n'; c=getchar() ) {if (c>='A' && c<='Z') { // 结点信息pT->nodes[newNode].data = c; //给结点赋值pT->nodes[newNode].firstchild = NULL; //给结点赋值pT->n++; //结点数+1//newNode有爸爸,即parentif (parent!=-1) {//创建孩子结点p = (CNode *)malloc(sizeof(CNode));if (!p) exit(OVERFLOW);p->index = newNode;p->next = NULL;//儿子父亲相认if (pT->nodes[parent].firstchild==NULL) {pT->nodes[parent].firstchild = p;} else {for (q=pT->nodes[parent].firstchild; q->next; q=q->next) ;q->next = p;}}} else if (c=='(') { //是newNode的孩子CreateCTreeByGList(pT, newNode); //开始创建newNode的孩子} else if (c==',') { //是newNode的兄弟,即parent的下一个孩子CreateCTreeByGList(pT, parent); //parent的下一个孩子return OK; //newNode结点构造完成(自己创建了、孩子创建了、兄弟创建了)} else if (c==')') { //parent构造完毕return OK; } else {return ERROR; //格式错误}}return OK;
}/*-------------------------|6.76 以广义表的形式输出 |-------------------------*/
Status PrintAsGList(CTree T,int parent) {CNode *p;if (T.n<=0) return ERROR; visit(T.nodes[parent].data);if (T.nodes[parent].firstchild) {printf("(");for (p=T.nodes[parent].firstchild; p; p=p->next) {PrintAsGList(T, p->index);if (p->next) printf(",");}printf(")");}return OK;
}int main() {
/*6.75测试数据
A(B(E,F),C(G),D)
A
A(B)
A(B,C)
A(B,C(D,E))
*/CTree T;T.n=0;T.r=0;CreateCTreeByGList(&T, -1); //6.75PrintAsGList(T, T.r); //6.76return 0;
}

这篇关于[树] △ 由广义表构造树(孩子链表CTree)并以广义表的形式输出(严蔚敏《数据结构》6.75|6.76 )的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

python多种数据类型输出为Excel文件

《python多种数据类型输出为Excel文件》本文主要介绍了将Python中的列表、元组、字典和集合等数据类型输出到Excel文件中,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一.列表List二.字典dict三.集合set四.元组tuplepython中的列表、元组、字典

Spring AI集成DeepSeek实现流式输出的操作方法

《SpringAI集成DeepSeek实现流式输出的操作方法》本文介绍了如何在SpringBoot中使用Sse(Server-SentEvents)技术实现流式输出,后端使用SpringMVC中的S... 目录一、后端代码二、前端代码三、运行项目小天有话说题外话参考资料前面一篇文章我们实现了《Spring

Rust格式化输出方式总结

《Rust格式化输出方式总结》Rust提供了强大的格式化输出功能,通过std::fmt模块和相关的宏来实现,主要的输出宏包括println!和format!,它们支持多种格式化占位符,如{}、{:?}... 目录Rust格式化输出方式基本的格式化输出格式化占位符Format 特性总结Rust格式化输出方式

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要

使用TomCat,service输出台出现乱码的解决

《使用TomCat,service输出台出现乱码的解决》本文介绍了解决Tomcat服务输出台中文乱码问题的两种方法,第一种方法是修改`logging.properties`文件中的`prefix`和`... 目录使用TomCat,service输出台出现乱码问题1解决方案问题2解决方案总结使用TomCat,

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

Python使用Colorama库美化终端输出的操作示例

《Python使用Colorama库美化终端输出的操作示例》在开发命令行工具或调试程序时,我们可能会希望通过颜色来区分重要信息,比如警告、错误、提示等,而Colorama是一个简单易用的Python库... 目录python Colorama 库详解:终端输出美化的神器1. Colorama 是什么?2.

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig