1 / 40

树和森林的概念 二叉树 (Binary Tree) 二叉树的表示 二叉树遍历 (Binary Tree Traversal) 堆 ( Heap ) 树与森林 (Tree & Forest)

第六章 树与森林. 树和森林的概念 二叉树 (Binary Tree) 二叉树的表示 二叉树遍历 (Binary Tree Traversal) 堆 ( Heap ) 树与森林 (Tree & Forest) 二叉树的计数 霍夫曼树 (Huffman Tree). 树和森林的概念. 树的定义 树是由 n ( n  0) 个结点组成的有限集合。如果 n = 0 ,称为空树;如果 n > 0 ,则  有一个特定的称之为 根 (root) 的结点,它只有直接后继,但没有直接前驱;

Download Presentation

树和森林的概念 二叉树 (Binary Tree) 二叉树的表示 二叉树遍历 (Binary Tree Traversal) 堆 ( Heap ) 树与森林 (Tree & Forest)

An Image/Link below is provided (as is) to download presentation Download Policy: Content on the Website is provided to you AS IS for your information and personal use and may not be sold / licensed / shared on other websites without getting consent from its author. Content is provided to you AS IS for your information and personal use only. Download presentation by click this link. While downloading, if for some reason you are not able to download a presentation, the publisher may have deleted the file from their server. During download, if you can't get a presentation, the file might be deleted by the publisher.

E N D

Presentation Transcript


  1. 第六章 树与森林 树和森林的概念 二叉树 (Binary Tree) 二叉树的表示 二叉树遍历 (Binary Tree Traversal) 堆( Heap ) 树与森林 (Tree & Forest) 二叉树的计数 霍夫曼树 (Huffman Tree)

  2. 树和森林的概念 树的定义 树是由n (n 0)个结点组成的有限集合。如果n = 0,称为空树;如果n > 0,则  有一个特定的称之为根(root)的结点,它只有直接后继,但没有直接前驱;  除根以外的其它结点划分为m (m 0)个互不相交的有限集合T0, T1, …, Tm-1,每个集合又是一棵树,并且称之为根的子树(subTree)。每棵子树的根结点有且仅有一个直接前驱,但可以有0个或多个直接后继。

  3. 结点(node) 结点的度(degree) 分支(branch)结点 叶(leaf)结点 子女(child)结点 双亲(parent)结点 兄弟(sibling)结点 祖先(ancestor)结点 子孙(descendant)结点 结点所处层次(level) 树的高度(depth) 树的度(degree) • 有序树 • 无序树 • 森林

  4. 树的抽象数据类型 template <class Type> class Tree { public: Tree ( ); ~Tree ( ); position Root ( ); BuildRoot ( const Type&value ); position FirstChild ( position p ); position NextSibling ( position p, position v ); position Parent ( position p ); TypeRetrieve ( position p ); position InsertChild ( constposition p, const Type &value ); position DeleteChild ( position p, inti ); void DeleteSubTree ( position t ); intIsEmpty ( ); }

  5. 二叉树 (Binary Tree) 二叉树的定义 一棵二叉树是结点的一个有限集合,该集合或者为空,或者是由一个根结点加上两棵分别称为左子树和右子树的、互不相交的二叉树组成。 二叉树的五种不同形态

  6. 二叉树的性质 性质1若二叉树的层次从0开始, 则在二叉树的第 i 层最多有 2i 个结点。(i 0) [证明用数学归纳法] 性质2高度为k的二叉树最多有 2k+1-1个结点。 (k -1) [证明用求等比级数前k项和的公式] 性质3对任何一棵二叉树, 如果其叶结点个数为 n0, 度为2的非叶结点个数为 n2, 则有 n0=n2+1

  7. 证明:若设度为1的结点有n1个,总结点个数为n,总边数为e,则根据二叉树的定义,证明:若设度为1的结点有n1个,总结点个数为n,总边数为e,则根据二叉树的定义, n = n0 + n1 + n2e = 2n2 + n1 = n - 1 因此,有 2n2 + n1= n0 + n1 + n2 - 1 n2 = n0 - 1 n0 = n2 + 1 定义1满二叉树(Full Binary Tree) 定义2完全二叉树(Complete Binary Tree) 若设二叉树的高度为h,则共有h+1层。除第h层外,其它各层(0h-1)的结点数都达到最大个数,第h层从右向左连续缺若干结点,这就是完全二叉树。

  8. 性质4具有n个结点的完全二叉树的高度为 log2(n+1)-1 证明:设完全二叉树的高度为h,则有 2h- 1 < n 2h+1- 1 2h < n+1  2h+1 取对数 h < log2(n+1)  h+1

  9. 性质5如果将一棵有n个结点的完全二叉树自顶向下,同一层自左向右连续给结点编号0, 1, 2, …, n-1,然后按此结点编号将树中各结点顺序地存放于一个一维数组中, 并简称编号为i的结点为结点i (0 i n-1。则有以下关系: • 若i == 0, 则 i 无双亲 若i > 0, 则 i 的双亲为(i -1)/2 • 若2*i+1 < n, 则 i 的左子女为2*i+1 若2*i+2 < n, 则 i 的右子女为2*i+2 • 若 i 为偶数, 且i != 0, 则其左兄弟为i-1 若 i 为奇数, 且i != n-1, 则其右兄弟为i+1 • i 所在层次为 log2 (i+1)

  10. 二叉树的抽象数据类型 template <class Type> classBinaryTree { public: BinaryTree ( ); BinaryTree ( BinTreeNode<Type> * lch, BinTreeNode<Type> * rch, Typeitem ); intIsEmpty ( ); BinTreeNode<Type> *Parent ( ); BinTreeNode<Type> *LeftChild ( ); BinTreeNode<Type> *RightChild ( ); intInsert ( const Type &item ); intFind ( const Type &item ) const; TypeGetData ( ) const; const BinTreeNode<Type> *GetRoot ( ) const; }

  11. 二叉树的表示 数组表示 完全二叉树的数组表示 一般二叉树的数组表示

  12. 由于一般二叉树必须仿照完全二叉树那样存储,可能会浪费很多存储空间,单支树就是一个极端情况。由于一般二叉树必须仿照完全二叉树那样存储,可能会浪费很多存储空间,单支树就是一个极端情况。 单支树

  13. 链表表示

  14. 二叉树链表表示的示例

  15. 二叉链表的静态结构

  16. 二叉树的类定义 template <class Type> classBinaryTree; template <class Type> ClassBinTreeNode { friend classBinaryTree; public: BinTreeNode ( ) :leftChild (NULL), rightChild (NULL) { } BinTreeNode ( Type item, BinTreeNode<Type> *left = NULL, BinTreeNode<Type> *right = NULL ) : data (item), leftChild (left), rightChild (right) { } const Type &GetData ( ) const { returndata;}

  17. BinTreeNode<Type> *GetLeft ( ) const { return leftChild;} BinTreeNode<Type> *GetRight ( ) const { returnrightChild;} voidSetData ( const Type &item ) { data = item;} voidSetLeft ( BinTreeNode <Type> *L ) { leftChild = L;} voidSetRight ( BinTreeNode <Type> *R ) {rightChild = R;} private: BinTreeNode<Type> *leftChild, *rightChild; Type data; };

  18. template <class Type> classBinaryTree { public: BinaryTree ( ) :root (NULL) { } BinaryTree ( Typevalue ) :RefValue (value), root (NULL) { } virtual ~BinaryTree ( ) {destroy ( root ); } virtualintIsEmpty ( ) { return root == NULL? 1 : 0;} virtualBinTreeNode <Type> *Parent ( BinTreeNode <Type> *current ) { returnroot == NULL||root == current? NULL: Parent ( root, current ); }

  19. virtual BinTreeNode <Type> *LeftChild ( BinTreeNode <Type> *current ) {return root != NULL? current→leftChild : NULL;} virtualBinTreeNode <Type> *RightChild ( BinTreeNode <Type> *current ) { returnroot != NULL? current→rightChild: NULL; } virtualintInsert ( const Type & item); virtualintFind ( const Type &item ) const; const BinTreeNode <Type> *GetRoot ( ) const { returnroot;} friend istream &operator >> ( istream &in, BinaryTree<Type>&Tree )

  20. friend ostream &operator << ( ostream &out, BinaryTree <Type> &Tree ) private: BinTreeNode <Type> *root; TypeRefValue; BinTreeNode <Type> *Parent ( BinTreeNode <Type> *start, BinTreeNode <Type> *current ); intInsert ( BinTreeNode<Type> * &current, const Type&item ) void Traverse ( BinTreeNode<Type> *current, ostream &out ) const intFind ( BinTreeNode<Type> *current, const Type &item ) const voiddestroy(BinTreeNode<Type> *current); }

  21. 二叉树部分成员函数的实现 template<class Type> voidBinaryTree<Type>:: destroy ( BinTreeNode<Type> *current ) { if ( current != NULL ) { destroy ( current→leftChild ); destroy ( current→rightChild ); delete current; } }

  22. Template <class Type> BinTreeNode <Type> *BinaryTree <Type>::Parent ( BinTreeNode <Type> *start, BinTreeNode <Type> *cuurent ) { if ( start == NULL ) returnNULL; if ( start→leftChild == current||start→rightChild == current ) return start; BinTreeNode <Type> *p; if ( ( p = Parent ( start→leftChild, current ) ) != NULL ) return p; else returnParent ( start→rightChild, current ); }

  23. Template <class Type> voidBinaryTree<Type>:: Traverse ( BinTreeNode <Type> *current, ostream &out ) const { if ( current != NULL ) { out << current→data << ‘ ’; Traverse ( current→leftChild, out ); Traverse ( current→rightChild, out ); } }

  24. Template <class Type> istream & operator >> ( istream &in, BinaryTree <Type> &Tree ) { Type item; cout << "构造二叉树:\n "; cout << "输入数据(用"<< Tree.RefValue << "结束): "; in >> item; while ( item != Tree.RefValue ) { Tree.Insert ( item ); cout << "输入数据(用"<< Tree.RefValue << "结束): "; in >> item; } returnin; }

  25. Template <class Type> ostream & operator << ( ostream &out, BinaryTree <Type> &Tree ) { out << “二叉树的前序遍历.\n"; Tree.Traverse ( Tree.root, out ); out << endl; returnout; }

  26. 6.3 遍历二叉树(Binary Tree Traversal) 所谓遍历二叉树,就是按某种次序访问树中的结点,要求每个结点访问一次且仅访问一次。 设访问根结点记作 V 遍历根的左子树记作 L 遍历根的右子树记作 R 则可能的遍历次序有 前序 VLR 镜像 VRL 中序 LVR 镜像 RVL 后序 LRV 镜像 RLV

  27. 中序遍历 (Inorder Traversal) 中序遍历二叉树算法的框架是: • 若二叉树为空,则空操作; • 否则 • 中序遍历左子树 (L); • 访问根结点 (V); • 中序遍历右子树 (R)。 遍历结果 a + b * c-d-e / f 表达式语法树

  28. 二叉树递归的中序遍历算法 emplate <class Type> voidBinaryTree <Type>::InOrder ( ) { InOrder ( root ); } template <class Type> void BinaryTree <Type>:: InOrder ( BinTreeNode <Type> *current ) { if ( current != NULL ) { InOrder ( current→leftChild ); cout << current→data; InOrder ( current→rightChild ); } }

  29. 中序遍历二叉树的递归过程图解

  30. 前序遍历 (Preorder Traversal) 前序遍历二叉树算法的框架是 • 若二叉树为空,则空操作; • 否则 • 访问根结点 (V); • 前序遍历左子树 (L); • 前序遍历右子树 (R)。 遍历结果 - + a * b-c d / e f

  31. 二叉树递归的前序遍历算法 template <class Type> voidBinaryTree <Type>::PreOrder ( ) { PreOrder ( root ); } template <class Type> void BinaryTree<Type>:: PreOrder ( BinTreeNode <Type> *current ) { if ( current != NULL ) { cout << current→data; PreOrder ( current→leftChild ); PreOrder ( current→rightChild ); } }

  32. 后序遍历 (Postorder Traversal) 后序遍历二叉树算法的框架是 • 若二叉树为空,则空操作; • 否则 • 后序遍历左子树 (L); • 后序遍历右子树 (R); • 访问根结点 (V)。 遍历结果 a b c d- * + e f / -

  33. 二叉树递归的后序遍历算法 template <class Type> void BinaryTree <Type>::PostOrder ( ) { PostOrder ( root ); } template <class Type> voidBinaryTree<Type>:: PostOrder ( BinTreeNode <Type> *current ) { if ( current != NULL ) { PostOrder ( current→leftChild ); PostOrder ( current→rightChild ); cout << current→data; } }

  34. 6.3.2 应用二叉树遍历的事例 1、利用二叉树后序遍历计算二叉树结点个数及二叉树的高度 template <class Type> int BinaryTree<Type>:: Size ( constBinTreeNode <Type> *t ) const { if ( t == NULL ) return 0; elsereturn 1 + Size ( t→leftChild ) + Size ( t→rightChild ); } template <class Type> intBinaryTree<Type>:: Height ( const BinTreeNode <Type> *t ) const {

  35. if ( t == NULL ) return-1; else return 1 + Max ( Height ( t→leftChild ), Height ( t→rightChild ) ); } 6.3 .3 遍历二叉树的非递归算法

  36. 1、利用栈的前序遍历非递归算法 Template <class type> void BinaryTree<Type>::PreOrdere () { Stack<Treenode<Type> * >st; BinTreeNode<type> * p=root; do{ while ( p!=NULL) { cout( p→data ); S.Push( p); p=p→leftchild; } if ( st.IsEmpty()=0 ){ p = S.getTop( ); st.Pop( ); p=p->rightchild} //进右子树 } }while (p!=NULL||!st.Isempty()) }

  37. 2、利用栈的中序遍历非递归算法 void InOrderTraverse ( Bitree T ) { StackTypeS;BitreeNode * p; S.makeEmpty( ); p = T;//初始化 do{ while ( p ) {S.Push(p);p = p→leftChild; } if ( !S.IsEmpty( ) ) { //栈非空 p = S.getTop( );S.Pop( );//退栈 printf ( p→data);//访问根结点 p = p→rightChild; //向右链走 } } while ( p || !S.IsEmpty( ) ); }

  38. 3、利用栈的后序遍历非递归算法 后序遍历时使用的栈的结点定义 typedef structStackNode{ enumtag{L, R}; BitreeNode * ptr; } StackNode; void PostOrderTraverse ( Bitree T ) { //后序遍历根为 T 的二叉树, 存储结构为二叉链表, //S是存储所经过二叉树结点的工作栈。其中, tag //是结点标记。当tag=L时, 表示刚才是在左子树//中, 从左子树退出时, 还要去遍历右子树。当tag //=R时, 表示刚才是在右子树中,在从右子树中退//出时, 去访问位于栈顶的结点。 ptrtag (LR)

  39. StackTypeS; BitreeNode * p; StackNodew; S.makeEmpty( ); p = T;//初始化 do { while ( p ) { w.ptr = p;w.tag = L;S.Push(w); p = p→leftChild; } //遍历左子树并把经过结点加左标记进栈 int continue = 1; //继续循环标记 while ( continue&& !S.IsEmpty( ) ) { w = S.getTop( );S.Pop( );p = w.ptr; switch (w.tag) {//判断栈顶的tag标记 caseL :w.tag = R;S.Push(w); continue = 0;//修改栈顶tag p = p→rightChild; break;

  40. caseR :printf (p→data); break; } } }while ( p|| !S.IsEmpty( ) ); }

More Related