当前位置:   article > 正文

二叉树:销毁_销毁二叉树

销毁二叉树

目的:使用C++模板设计并逐步完善二叉树的抽象数据类型(ADT)。

内容:(1)请参照链表的ADT模板,设计二叉树并逐步完善的抽象数据类型。(由于该环境目前仅支持单文件的编译,故将所有内容都集中在一个源文件内。在实际的设计中,推荐将抽象类及对应的派生类分别放在单独的头文件中。参考教材、课件,以及网盘中的链表ADT原型文件,自行设计二叉树的ADT。)

注意:二叉树ADT的基本操作的算法设计很多要用到递归的程序设计方法。

(2)基本操作6:在二叉树的二叉链表存储形式建立的基础上,设计二叉树的销毁算法,主要用于析构函数中。完成后将其加入到二叉树的ADT基本操作集中。

要求使用递归的程序设计方法,设计并完成二叉树的销毁算法。

初始条件:二叉树T存在。

操作结果:销毁二叉树T。

参考函数原型:

//销毁树(外壳部分,public)

template<class ElemType>

void BinaryTree<ElemType>::BinaryTreeDestroy();

//销毁树(递归部分。private)

template<class ElemType>

void BinaryTree<ElemType>::BinaryTreeDestroy_Cursive( BinaryTreeNode<ElemType> *T );

正片开始:

仍然用之前的模板上储存结构等代码,然后再按题目要求写两个函数,一个private,一个public。

代码如下:

  1. #include<iostream>
  2. #include<cstring>
  3. #include<vector>
  4. #include<queue>
  5. #include<sstream>
  6. using namespace std;
  7. /* 二叉表的结点定义 */
  8. template<class ElemType>
  9. struct BinaryTreeNode
  10. {
  11. ElemType data;
  12. BinaryTreeNode<ElemType> *LChild, *RChild;
  13. BinaryTreeNode() : LChild(NULL), RChild(NULL) {} //构造函数1,用于构造根结点
  14. BinaryTreeNode(const ElemType &item, BinaryTreeNode<ElemType> *Lptr = NULL, BinaryTreeNode<ElemType> *Rptr = NULL) //构造函数2,用于构造其他结点
  15. //函数参数表中的形参允许有默认值,但是带默认值的参数需要放后面
  16. {
  17. LChild = Lptr;
  18. RChild = Rptr;
  19. data = item;
  20. }
  21. ElemType getData()
  22. {
  23. return data; //取得结点中的数据
  24. }
  25. void SetLChild( BinaryTreeNode<ElemType> *link )
  26. {
  27. LChild = link; //修改结点的左孩子域
  28. }
  29. void SetRChild( BinaryTreeNode<ElemType> *link )
  30. {
  31. RChild = link; //修改结点的右孩子域
  32. }
  33. void SetData( ElemType value )
  34. {
  35. data = value; //修改结点的data
  36. }
  37. BinaryTreeNode<ElemType> * GetLChild() const
  38. {
  39. return LChild; //获取左孩子结点
  40. }
  41. BinaryTreeNode<ElemType> * GetRChild() const
  42. {
  43. return RChild; //获取左孩子结点
  44. }
  45. };
  46. template<class ElemType>
  47. bool visit(BinaryTreeNode<ElemType> *root,int &num)
  48. {
  49. if(!root)
  50. return false;
  51. else
  52. {
  53. if(num==0)
  54. {
  55. cout<<root->data;
  56. num++;
  57. }
  58. else
  59. cout<<","<<root->data;
  60. return true;
  61. }
  62. }
  63. //二叉树
  64. template<class ElemType>
  65. class BinaryTree
  66. {
  67. private:
  68. BinaryTreeNode<ElemType> *root; // 头指针
  69. //销毁树(递归准备,private)
  70. bool Destroy_Cursive( BinaryTreeNode<ElemType> *T )
  71. {
  72. if(T==NULL) return false;
  73. Destroy_Cursive(T->LChild);
  74. Destroy_Cursive(T->RChild);
  75. free(T);
  76. T=NULL;
  77. return true;
  78. }//按题目要求放在private里的函数
  79. public:
  80. //无参数的构造函数
  81. BinaryTree():root(NULL) {}
  82. //带参数的构造函数
  83. BinaryTree(const ElemType &item)
  84. {
  85. root = new BinaryTreeNode<ElemType>(item);
  86. }
  87. //生成树
  88. void makeBinaryTree( const ElemType &item, BinaryTree &left, BinaryTree &right);
  89. //拷贝构造函数
  90. //LinkQueue(LinkQueueList<ElemType> &Queue);
  91. //析构函数
  92. //~BinaryTree()
  93. //{
  94. // BinaryTreeDestroy();
  95. //}
  96. //重载函数:赋值
  97. //LinkList<ElemType>& operator=(LinkList<ElemType> &List);
  98. //销毁树
  99. void BinaryTreeDestroy();
  100. //销毁子树
  101. void ChildDestroy(int flag);
  102. //返回二叉树结点的个数
  103. int BinaryTreeSize( BinaryTreeNode<ElemType> *T ) const;
  104. //判断二叉树是否为空
  105. bool BinaryTreeisEmpty() const
  106. {
  107. return root == NULL;
  108. }
  109. //获取根结点元素值
  110. ElemType GetRootData() const
  111. {
  112. return root->data;
  113. }
  114. //bool Location(ElemType &x, BinaryTreeNode<ElemType> * &location);
  115. //设置根结点
  116. void SetRoot(BinaryTreeNode<ElemType> * p)
  117. {
  118. root = p;
  119. }
  120. //获取根结点
  121. BinaryTreeNode<ElemType> * GetRoot() const
  122. {
  123. return root;
  124. }
  125. //前序遍历
  126. bool Pre(BinaryTreeNode<ElemType> *T,bool(*visit)(BinaryTreeNode<ElemType> *T,int &num),int &num) const; //前序遍历(递归)//num的初始值为0,作用为控制输出格式(最后1个结点后不加“,”)
  127. //中序遍历
  128. bool In(BinaryTreeNode<ElemType> *T,bool(*visit)(BinaryTreeNode<ElemType> *T,int &num),int &num) const; //中序遍历(递归)
  129. //后序遍历
  130. bool Post(BinaryTreeNode<ElemType> *T,bool(*visit)(BinaryTreeNode<ElemType> *T,int &num),int &num) const; //后序遍历(递归)
  131. //层次遍历
  132. bool Ceng(BinaryTreeNode<ElemType> *T,bool(*visit)(BinaryTreeNode<ElemType> *T,int &num),int &num) const;//层次遍历 (递归)
  133. //建立二叉树的存储结构
  134. BinaryTreeNode<ElemType>* CreateBinaryTree(vector<ElemType> &x, ElemType &sym, int &n);
  135. bool Destroy( );//public这里还要放一个
  136. };
  137. //建立二叉树的存储结构 (递归部分,成员函数)
  138. template<class ElemType>
  139. BinaryTreeNode<ElemType>* BinaryTree<ElemType>::CreateBinaryTree(vector<ElemType> &x, ElemType &sym, int &n)
  140. {
  141. ElemType ch = x[n];
  142. n++;
  143. if (ch == sym)
  144. {
  145. return NULL;
  146. }
  147. else
  148. {
  149. BinaryTreeNode<ElemType> *Node = new BinaryTreeNode<ElemType>;
  150. Node->data = ch;
  151. Node->LChild = CreateBinaryTree(x, sym, n);
  152. Node->RChild = CreateBinaryTree(x, sym, n);
  153. return Node;
  154. }
  155. }
  156. //建立二叉树的存储结构 (外壳)
  157. template<class ElemType>
  158. void CreateTree(BinaryTree<ElemType> &T, ElemType &str, ElemType &sym)
  159. {
  160. ElemType tmp;
  161. vector<ElemType> t;
  162. stringstream input_T(str);
  163. while(input_T>>tmp)
  164. {
  165. t.push_back(tmp);
  166. }
  167. BinaryTreeNode<ElemType> *root;
  168. int num=0;
  169. root=T.CreateBinaryTree(t,sym,num);
  170. T.SetRoot(root);
  171. }
  172. template<class ElemType>
  173. void BinaryTree<ElemType>::makeBinaryTree(const ElemType &item, BinaryTree &left, BinaryTree &right)
  174. {
  175. root=new BinaryTreeNode<ElemType>(item,left.root,right.root);
  176. left.root=NULL;
  177. right.root=NULL;
  178. }
  179. template<class ElemType>
  180. bool BinaryTree<ElemType>::Destroy()//作为成员函数出现的,前面仍然加了BinaryTree<ElemType>::
  181. {
  182. BinaryTreeNode<ElemType>*p=GetRoot();//取的类私有的里面的函数,所以直接=GetRoot()
  183. if(Destroy_Cursive( p ))//这里用的是类的私有函数,定义在private里面的,因为上一行的上一行里有BinaryTree<ElemType>::,就直接用啦
  184. {
  185. return true;
  186. }//然后我的一个问题就是之前写三个遍历的时候把函数名都取的一样的,然后后面自己也没分清楚,结果写这个题的时候,
  187. else return false;//因为题目里面要求一个放在私有的private里,一个放在公有的public里
  188. } //然后都用一样的名字的话就会出现重复定义的问题(其实参数形式不一样的话也顶多重载,但我开始写的时候在探索过程出了点小差错
  189. //把公有的这个函数也带上了参数,然后就出现了重复定义的错误)
  190. int main()
  191. {
  192. BinaryTree<string>BT;
  193. string mytree;
  194. string sym;
  195. getline(cin,sym);
  196. getline(cin,mytree);
  197. CreateTree(BT,mytree,sym);
  198. if(BT.Destroy())//用的内部函数,BT.函数 即可
  199. {
  200. cout<<"success"<<endl;
  201. return 0;
  202. }
  203. else
  204. cout<<"fail"<<endl;//也可以用判空的方法,这里这样直接看有没有destroy干净,没有的话用个else输出fail即可
  205. return 0;
  206. }

谢谢支持!在学习ADT型,会继续加油的!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/盐析白兔/article/detail/622012
推荐阅读
相关标签
  

闽ICP备14008679号