1、4 实验一 基于二叉链表的二叉树的实现4.1 问题描述基于二叉链表和队列及其堆栈存储结构,实现二叉链表的二叉树的对数据进行各种必要的操作。4.2 系统设计1.2.1 提供 20 个功能,分别是:1.2.2 二叉链表的结构试一堆栈和队列的形式进行储存的分别是:1.2.3 在程序中所定义的数据结构有:2.3 系统实现1.3.1 InitTree 功能初始二叉链表,传入的是头结点地址。申请一个存储空间,并用头结点中的首结点指针指向该空间首地址,相应的 时间复杂度为 1。具体实现如下:1.3.2 DestroyTree 功能销毁头结点中首结点址针指向的线性存储空间,传入的是头结点地址。具体实现如下:1
2、.3.3 CreateBiTree 功能与 DestroyBiTree 类似但是又有不同, ClearBiTree 并不销毁物理空间,而是修改逻辑关系值:1.3.4 ClearBiTree 功能与 DestroyBiTree 类似但是又有不同, ClearBiTree 并不销毁物理空间,而是修改逻辑关系值1.3.5 BiTreeEmpty 功能判空功能,判断表是否为空表。时间复杂度为 1,因为只需判断一次就可以知道是否为空。实现如下:1.3.6 BiTreeDepth 功能求二叉链表深度的功能,由于创建过程中已经把表长信息包含在头结点中,所以直接调用并显示即可1.3.7 Root(BiTree
3、 T)功能获取二叉链表的根节点的元素,通过遍历二叉链表中的元素,来逐个判断,时间复杂度为(n) 。1.3.8 Value(BiTree T,TElemType e)功能求指定元素的前一个元素的内容,传入头结点值、包含指定元素信息的一个临时表结点值、存储前一个元素的表结点地址。主要思路是递归算法。时间复杂度为 O(n)。具体实现如下:1.3.9 Assign 功能求指定元素的后一个元素的内容,传入头结点值、包含指定元素信息的一个临时表结点值、存储前一个元素的表结点地址。找到后,把新的数据值赋给所找到的节点。时间复杂度为 O(n)。具体实现如下:1.3.10 Parent 功能找双亲节点,找到后输
4、出1.3.11 LeftChild 功能查找左孩子,利用递归的算法,与遍历的时间复杂度为相同 O(n)1.3.12RightChild 功能查找右孩子,利用递归的算法,与遍历的时间复杂度为相同 O(n)1.3.13 LeftSibling 功能查找节点的左边的堂兄弟的,找到后输出该节点的数据1.3.14 RightSibling 功能查找节点的右边的堂兄弟的,找到后输出该节点的数据1.3.15 InsertChild 函数在二叉链表中插入新的节点1.3.15 DeleteChild 功能删除指定编号的数据元素,传入头结点地址、编号 i、表结点类型结构体地址来返回被删除元素内容。执行前先判断传入
5、的编号是否在可寻找范围内。执行删除操作之后,进行“移位”运算。时间复杂度仍为 O(n)。如下:1.3.16 PreOrderTraverse 功能前序遍历二叉链表中的数据,采用先遍历左孩子,再访问根节点,后访问右孩子的思想来实现前序遍历的算法的。1.3.17 InOrderTraverse 功能中序遍历的函数,对二叉链表的数据进行访问,并且利用 PreOrderTraverse 函数1.3.18 PostOrderTraverse 功能采用后续遍历的思想,利用先序遍历的函数进行1.3.19 LevelOrderTraverse 功能完全遍历二叉链表中的数据,并进行输出的1.3.20 Point
6、 功能定位节点的函数,在需要查找二叉链表二叉树的节点的时候,可以直接调用该函数,进行处理,相应的代码如下1.3.21 FILE * fileOpen 功能读取功能,通过 fscanf 实现格式化读取,同时结合 CreateList 函数实现顺序1.3.22 BiTNode * Create(FILE *fp)功能把二叉链表二叉树的数据写入到文件中去1.4 效率分析在上面介绍各功能时已经提到时间复杂度的计算了,这里再简单分析一下。具有同数量级复杂度的功能在实现方法上一般近似。比如 InOrderTraverse,PostOrderTraverse,BiTreeDepth,LevelOrderTr
7、averse它们都是基于 PreOrderTraverse来设计的,所以效率都是 O(n);而 Root,Value ,Assign,Parent ,LeftChild,RightChild,LeftSiblingRightSibling,InsertChild , DeleteChild是基于 VisitPoint,平均效率为 O(n); InitTree DestroyBiTree 所需信息,所以效率为 O(1);CreateBiTreeClearBiTreeBiTreeEmpty 都要对二叉链表,平均效率为 O(n)。实验总结与评价我做了这个实验发现自己的编程能力很不好,自己的脑袋中有相
8、应的想法和主意,但是因为自己的编程能力很不好也就实现不了自己的想法。二叉链表的二叉树的时候,实现二叉链表线性的对我来说还可以实现,因为线性的所用到方法和技术,在学习十字链表的时候练习的比较少,实现起来难度是很大。特别是有了老师给的框架以后,我们要做的任务就是向里面填我们自己写的函数,在填写的过程中,我深深的感受到了,认真的重要性,因为我在写好调试的中发现了很多,因为自己的不小心和在敲代码的过程中的不认真而造成的很不应该的错误,这些错误也给自己在调试的过程中也造成了很大的麻烦,因为是不认真而犯的错误,因此调试的过程中也很不好发现。对我来说,因为我的 C 语言的功底很不好,运用指针和链表的能力还没
9、有能达到运用自如,理解深刻的地步,所以在顺序链表的链表的实现中,对我来说是一个很大的挑战,我有很多不会的地方通过自己看书,问室友和上网查询,一点一点的写了出来,肯定现在还是会有很多的问题,但是这也是我一直在努力把它做的更好,在调试的中出现了很多的 BUG,自己一个个的慢慢的消除掉了,做出了,现在的程序。如果问自己的体会,那一定是希望我自己以后多多的动手,把以前 C 语言的书好好再复习一遍,还有就是把现在正在学习的数据结构的书上各个程序,自己要一个个的敲一遍,练习一下自己的熟悉程度。总的来说,我对这次的实验是很有感触的。因为,这次实验让我认识到了,自己的编程能力的低下,如果自己再不下一下功夫的话
10、,那么数据结构的考试自己就十分的危险了。因此,我要加紧复习 C 语言的知识和数据结构学过的内容,争取自己能在接下来的学习中能有些进步。附录:参考书数据结构 (C 语言版)严蔚敏 吴伟民编著C 语言程序设计 曹计昌,李开编著实验心得体会对于这两次的实验,我自己的体会是很深刻的,也是记忆深刻的。因为,正是因为这两次的实验深深地让我认识到了自己的水平是多么的低下,以前,自己还有点夜郎自大的认为,自己对所学的东西,自己掌握的还差不多了呢。但是,经过这次的实验,我真的是清楚的发现自己对所学的知识的掌握还差的很多,自己还有很多的功课要补。第一,以前无论是学习 C 语言还是数据结构,我的方法是拿着书本看,还
11、有就是拿着练习本写一写,而自己家上机的实践的时间是非常少的,因为我感觉上机得到的结构一定会和自己想的和写的一样呢,显然,我是错误的,因为在这次的实验里我就发现,即使是书上一模一样的代码,在机子上也是有很大 的可能出错的,更不用说是自己写的了,在写线性表,线性链表和二叉链表的时候,我出现了用书上的代码不能用的情况,而且是非常严重的错误。有些声明和指针的问题会出现很大的不同。我的体会是,从现在起,重视上机的过程,多书上的程序一定要在机子上跑一下,然后再分析一下,出现这种结果的原因和整个程序的流程。第二,就是实验的 时候的规范的问题,由于,自己写代码没有很好的习惯和规则,于是,在自己写好的程序出现错
12、误后自己不能够很快的 找到出现错误的位置,比如,对全局变量声明的时候,全局变量的位置问题,在结构和联合声明指针的时候,指针的形式和指针的命名的形式问题,这些错误都有在自己的实验的过程中出现,而且,也给自己带来了很大的麻烦。我的体会是,以后再写程序的时候一定遵守一定的规则和习惯,例如关键词的命名习惯,指针的使用形式和结构联合中的一些形式的问题,应该遵循一定的规则和习惯,因为,只有这样的自己在写好的调试和检查的过程中才不会走那么多 的弯路,才会把做事的速度提高上去。最后,就是自己的一些心得体会对这次的实验。做什么事情都要认真对待,无论事情的大小,因为只有这样自己才会养成认真做事的习惯,这次的数据结
13、构的实验让我深深的意识到了这一点。附录:实验三代码:#include #include #include #include#include #define LIST_INIT_SIZE 100#define LISTINCREMENT 10#define TRUE 1#define FALSE 0#define OK 1#define ERROR 0#define INFEASTABLE -1#define OVERFLOW -2#define MAX_TREE_SIZE 100typedef int Status;typedef int TElemType; /数据元素类型定义,注意这里是整
14、型的,可以使chartypedef TElemType SqBiTreeMAX_TREE_SIZE;SqBiTree bt;typedef structTElemType *base;TElemType *top;int stacksize;SqStack;Status InitStack(SqStack*S);Status Pop(SqStack*S,TElemType e);Status Push(SqStack*S,TElemType e);Status StackEmpty(SqStack*S);/全局变量的声明char *m_pCharBuf = NULL;int m_nList10
15、0;int m_nCount = 0;char* openFileOnlyRead(char * fileName);void writeQuickSortResult(char *pResult);char* openFileOnlyRead(char * fileName)int nLen = 0;FILE *pFile = fopen(fileName, “r“); /打开文件fseek(pFile, 0, SEEK_END); /文件指针移到文件尾nLen = ftell(pFile); /得到当前指针位置, 即是文件的长度rewind(pFile); /文件指针恢复到文件头位置/动态
16、申请空间, 为保存字符串结尾标志0, 多申请一个字符的空间m_pCharBuf = (char*)malloc(sizeof(char)* nLen + 1);if (!m_pCharBuf)perror(“内存不够!n“);exit(0);/读取文件内容/读取的长度和源文件长度有可能有出入,这里自动调整 nLennLen = fread(m_pCharBuf, sizeof(char), nLen, pFile);m_pCharBufnLen = 0; /添加字符串结尾标志/printf(“%sn“, pchBuf); /把读取的内容输出到屏幕fclose(pFile); /关闭文件/fre
17、e(pchBuf); /释放空间return m_pCharBuf;/写入排序完成后的结果void writeQuickSortResult(char *pResult)FILE *pFile = fopen(“QuickSortResult.txt“, “w“); /打开文件fputs(pResult, pFile); /写入数据fclose(pFile); /关闭文件typedef struct BiTNodeTElemType data;struct BiTNode *lchild,*rchild;BiTNode,*BiTree;typedef BiTree QElemType;type
18、def struct QNodeQElemType data;struct QNode *next;QNode,*QueuePtr;typedef struct LinkQueue QueuePtr front,rear;LinkQueue;void InitTree(BiTree*T);void DestroyBiTree(BiTree *T);void CreateBiTree(BiTree *T);Status ClearBiTree(BiTree T);Status BiTreeEmpty(BiTree T);Status BiTreeDepth(BiTree T);Status Ro
19、ot(BiTree T);Status Value(BiTree T,TElemType e);Status Assign(BiTree T,TElemType e,int value);Status Parent(BiTree T,TElemType e);Status LeftChild(BiTree T,TElemType e);Status RightChild(BiTree T,TElemType e);Status LeftSibling(BiTree T,TElemType e);Status RightSibling(BiTree T,TElemType e);Status I
20、nsertChild(BiTree T,int LR,BiTree C);Status DeleteChild(BiTree T,int LR);Status PreOrderTraverse(BiTree T,Status(*Visit)(TElemType e);Status InOrderTraverse(BiTree T,Status(*Visit)(TElemType e);Status PostOrderTraverse(BiTree T,Status(*Visit)(TElemType e);Status LevelOrderTraverse(BiTree T,Status(*V
21、isit)(TElemType e);Status Visit(TElemType e);BiTree Point(BiTree T,TElemType s); /返回二叉树中指向元素值为s的结点的指针void InitQueue(LinkQueue Q);/构造一个空队列Status QueueEmpty(LinkQueue Q);/判断队列是否为空void EnQueue(LinkQueue Q,QElemType e);/插入元素为新的队尾元素Status DeQueue(LinkQueue Q,QElemType e);/删除队头元素BiTNode * Create(FILE *fp)
22、;FILE * fileOpen();void main(void)int i;/文件内容读取出来char *pText = openFileOnlyRead(“resource.txt“);printf(“%snn“, pText);BiTree T;FILE *p;TElemType e;int n;int value;int op=1;while(op)system(“cls“);printf(“nn“);printf(“ Menu for Linear Table On Sequence Structure n“);printf(“-n“);printf(“ 1. InitTree 1
23、1. LeftChildn“);printf(“ 2. DestroyBiTree 12. RightChildn“);printf(“ 3. CreateBiTree 13. LeftSiblingn“);printf(“ 4. ClearBiTree 14. RightSiblingn“);printf(“ 5. BiTreeEmpty 15. InsertChildn“);printf(“ 6. BiTreeDepth 16. DeleteChildn“);printf(“ 7. Root 17. PreOrderTraversen“);printf(“ 8. Value 18. InO
24、rderTraversen“);printf(“ 9. Assign 19. PostOrderTraversen“);printf(“ 10. Parent 20. LevelOrderTraversen“);printf(“ 0. Exitn“);printf(“-n“);printf(“ 请选择你的操作020:“);scanf(“%d“,switch(op)case 1:InitTree(BiTNode * Create(p);FILE * fileOpen();if(!(T)=NULL)printf(“n-二叉树初始化成功!n“);elseprintf(“二叉树创建失败!n“);get
25、char();getchar();break;case 2:printf(“是否要销毁二叉树!(1为是,0是否)n“);scanf(“%d“,if(n=1)DestroyBiTree(else return 0;if(T!=NULL)printf(“n-二叉树成功销毁实现!n“);elseprintf(“n-DestroyList功能待实现“);getchar();getchar();break;case 3:/InitTree(printf(“Please input the char:e=n“);scanf(“%d“,CreateBiTree(T);if(T)!=NULL)printf(“
26、n-CreateBiTree功能实现!n“);elseprintf(“n-CreateBiTree功能待实现!n“);getchar();getchar();break;case 4:ClearBiTree(T);if(T)printf(“n-ClearBiTree功能待实现!n“);elseprintf(“n-ClearBiTree功能实现!n“);0-getchar();getchar();break;case 5:BiTreeEmpty(T);if(T)printf(“n-BiTreeEmpty功能实现!n“);elseprintf(“n-BiTreeEmpty功能待实现!n“);get
27、char();getchar();break;case 6:BiTreeDepth(T);if(T)printf(“n-BiTreeDepth功能实现!n“);elseprintf(“n-BiTreeDepth功能待实现!n“);getchar();getchar();break;case 7:Root(T);if(T)printf(“n-Root功能实现!n“);elseprintf(“n-Root功能待实现!n“);getchar();getchar();break;case 8:printf(“Please input the node of you want:e=n“);scanf(“
28、%c“,Value(T,e);if(T=NULL)printf(“n-Value功能实现!n“);elseprintf(“n-Value功能待实现!n“);getchar();getchar();break;case 9:printf(“Please input the node and number of you want:e=nvalue=n“);scanf(“%c%d“,Assign(T,e,value);if(T=NULL)printf(“n-Assign功能实现!n“);elseprintf(“n-Assign功能待实现!n“);getchar();getchar();break;c
29、ase 10:printf(“Please input the node of you want:e=n“);scanf(“%c“,Parent(T,e);if(T=NULL)printf(“n-Parent功能实现!n“);elseprintf(“n-Parent功能待实现!n“);getchar();getchar();break;case 11:printf(“Please input the node of you want:e=n“);scanf(“%c“,LeftChild(T,e);if(T!=NULL)printf(“n-LeftChild功能实现!n“);elseprintf
30、(“n-LeftChild功能待实现n“);getchar();getchar();break;case 12:printf(“Please input the node of you want:e=n“);scanf(“%c“,RightChild(T,e);if(T!=NULL)printf(“n-RightChild功能实现n“);elseprintf(“n-RightChild功能待实现!n“);getchar();getchar();break;case 13:printf(“Please input the node of you want:e=n“);scanf(“%c“,Lef
31、tSibling(T,e);if(T!=NULL)printf(“n-LeftSibling功能实现!n“);elseprintf(“n-LeftSibling功能待实现n“);getchar();getchar();break;case 14:printf(“Please input the node of you want:e=n“);scanf(“%c“,RightSibling(T,e);if(T!=NULL)printf(“n-LeftSibling功能实现!n“);elseprintf(“n-LeftSibling功能待实现!n“);getchar();getchar();brea
32、k;case 15:/printf(“Please input the node of you want:p,e,LR and C=n“);/ scanf(“%c“,/ InsertChild(T,P,LR,C);printf(“线性表是空表!n“);getchar();getchar();break;case 16:printf(“线性表是空表!n“);getchar();getchar();break;case 17:printf(“线性表是空表!n“);getchar();getchar();break;case 18:printf(“线性表是空表!n“);getchar();getch
33、ar();break;case 19:printf(“Please input the node of you want:e=n“);scanf(“%c“,PostOrderTraverse(T,e);if(T!=NULL)printf(“功能实现了!n“);elseprintf(“功能待实现了!n“);getchar();getchar();break;case 20:printf(“线性表是空表!n“);getchar();getchar();break;case 0:break;/end of switch/end of whilechar result1000 = “QuickSortResult:“ ;for(i = 0; i data=NULL;return OK;void DestroyBiTree(BiTree *T)if(T!=NULL)DestroyBiTree(*T)-lchild);DestroyBiTree(*T)-rchild);free(T);T=NULL;return OK;