当前位置:文档之家› 树二叉树和森林

树二叉树和森林

树二叉树和森林
树二叉树和森林

第6章树与森林

6-1 写出用广义表表示法表示的树的类声明,并给出如下成员函数的实现:

(1) operator >> ( ) 接收用广义表表示的树作为输入,建立广义表的存储表示;

(2) 复制构造函数用另一棵表示为广义表的树初始化一棵树;

(3) operator == ( ) 测试用广义表表示的两棵树是否相等;

(4) operator << ( ) 用广义表的形式输出一棵树;

(5) 析构函数清除一棵用广义表表示的树。

【解答】

#include

#define maxSubTreeNum 20; //最大子树(子表)个数

class GenTree;//GenTree类的前视声明

class GenTreeNode { //广义树结点类的声明

friend class GenTree;

private:

int utype; //结点标志:=0, 数据; =1, 子女

GenTreeNode * nextSibling; //utype=0, 指向第一个子女;

//utype=1或2, 指向同一层下一兄弟union { //联合

char RootData; //utype=0, 根结点数据

char Childdata;//utype=1, 子女结点数据

GenTreeNode *firstChild; //utype=2, 指向第一个子女的指针}

public:

GenTreeNode ( int tp, char item ) : utype (tp), nextSibling (NULL)

{ if ( tp == 0 ) RootData = item; else ChildData = item; }

//构造函数:构造广义表表示的树的数据结点

GenTreeNode (GenTreeNode *son = NULL ) : utype (2), nextSibling (NULL), firstChild ( son ) { }

//构造函数:构造广义表表示的树的子树结点

int nodetype ( ) { return utype;}//返回结点的数据类型

char GetData ( ) { return data;} //返回数据结点的值

GenTreeNode * GetFchild ( ) { return firstChild; } //返回子树结点的地址

GenTreeNode * GetNsibling ( ) { return nextSibling; } //返回下一个兄弟结点的地址

void setInfo ( char item ) { data = item; } //将结点中的值修改为item

void setFchild ( GenTreeNode * ptr ) { firstChild = ptr; }//将结点中的子树指针修改为ptr

void setNsinbilg ( GenTreeNode * ptr ) { nextSibling = ptr; }

};

class GenTree {//广义树类定义

private:

GenTreeNode *first; //根指针

char retValue;//建树时的停止输入标志

GenTreeNode *Copy ( GenTreeNode * ptr ); //复制一个ptr指示的子树

void Traverse ( GenListNode * ptr );//对ptr为根的子树遍历并输出void Remove ( GenTreeNode *ptr ); //将以ptr为根的广义树结构释放friend int Equal ( GenTreeNode *s, GenTreeNode *t ); //比较以s和t为根的树是否相等public:

GenTree ( ); //构造函数

GenTree ( GenTree& t );//复制构造函数

~GenTree ( ); //析构函数

friend int operator == ( GenTree& t1, GenTree& t2 );//判两棵树t1与t2是否相等friend istream& operator >> ( istream& in, GenTree& t );//输入

friend ostream& operator << ( ostream& out, GenTree& t ); //输出

}

(1) operator >> ( ) 接收用广义表表示的树作为输入,建立广义表的存储表示

istream& operator >> ( istream& in, GenTree& t ) {

//友元函数, 从输入流对象in接受用广义表表示的树,建立广义表的存储表示t。

t.ConstructTree ( in, retValue );

return in;

}

void GenTree :: ConstructTree ( istream& in, char& value ) {

//从输入流对象in接受用广义表表示的非空树,建立广义表的存储表示t。

Stack st (maxSubTreeNum);//用于建表时记忆回退地址GenTreeNode * p, q, r;Type ch;

cin >> value; //广义树停止输入标志数据cin >> ch; first = q = new GenTreeNode ( 0, ch ); //建立整个树的根结点

cin >> ch;if( ch == ‘(’ ) st.Push ( q );//接着应是‘(’, 进栈

cin >> ch;

while ( ch != value ) {//逐个结点加入switch ( ch ) {

case‘(’ : p = new GenTreeNode ( q );//建立子树, p->firstChild = q r = st.GetTop( );st.Pop( );//从栈中退出前一结点

r->nextSibling = p;//插在前一结点r之后

st.Push( p );st.Push( q ); //子树结点及子树根结点进栈

break;

case‘)’ : q->nextSibling = NULL;st.pop( );//子树建成, 封闭链, 退到上层if ( st.IsEmpty ( ) == 0 )q = st.GetTop( ); //栈不空, 取上层链子树结点

else return 0;//栈空, 无上层链, 算法结束

break;

case‘,’ :break;

default : p = q;

if ( isupper (ch) ) q = new GenTreeNode ( 0, ch ); //大写字母, 建根结点

else q = new GenTreeNode ( 1, ch );//非大写字母, 建数据结点

p->nextSibling = q;//链接

}

cin >> ch;

}

}

(2) 复制构造函数用另一棵表示为广义表的树初始化一棵树;

GenTree :: GenTree ( const GenTree& t ) {//共有函数first = Copy ( t.first );

}

GenTreeNode* GenTree :: Copy ( GenTreeNode *ptr ) {

//私有函数,复制一个ptr指示的用广义表表示的子树

GenTreeNode *q = NULL;

if ( ptr != NULL ) {

q = new GenTreeNode ( ptr->utype, NULL );

switch ( ptr->utype ) {//根据结点类型utype传送值域case 0 : q->RootData = ptr->RootData;break; //传送根结点数据

case 1 : q->ChildData = ptr->ChildData;break; //传送子女结点数据

case 2 : q->firstChild = Copy ( ptr->firstChild );break; //递归传送子树信息

}

q->nextSibling = Copy ( ptr->nextSibling ); //复制同一层下一结点为头的表}

return q;

}

(3) operator == ( ) 测试用广义表表示的两棵树是否相等

int operator == ( GenTree& t1, GenTree& t2 ) {

//友元函数:判两棵树t1与t2是否相等, 若两表完全相等, 函数返回1, 否则返回0。

return Equal ( t1.first, t2.first );

}

int Equal ( GenTreeNode *t1, GenTreeNode *t2 ) {

//是GenTreeNode的友元函数

int x;

if ( t1 == NULL && t2 == NULL ) return 1; //表t1与表t2都是空树, 相等if ( t1 != NULL && t2 != NULL && t1->utype == t2->utype ) { //两子树都非空且结点类型相同switch (t1->utype ) { //比较对应数据

case 0 : x =( t1->RootData == t2->RootData ) ? 1 : 0;//根数据结点

break;

case 1 :x = ( t1->ChildData == t2->ChildData ) ? 1 : 0;//子女数据结点

break;

case 2 : x = Equal ( t1->firstChild, t2->firstChild ); //递归比较其子树

}

if ( x ) return Equal ( t1->nextSibling, t2->nextSibling );

//相等, 递归比较同一层的下一个结点; 不等, 不再递归比较}

return 0;

}

(4) operator << ( ) 用广义表的形式输出一棵树

ostream& operator << ( ostream& out, GenTree& t ) {

//友元函数, 将树t输出到输出流对象out。

t.traverse ( out, t.first );

return out;

}

void GenTree :: traverse ( ostream& out, GenTreeNode * ptr ) {

//私有函数, 广义树的遍历算法

if ( ptr != NULL ) {

if ( ptr->utype == 0 )out << ptr->RootData << ‘(’;//根数据结点

else if ( ptr->utype == 1 ) {//子女数据结点out << ptr->ChildData;

if ( ptr->nextSibling != NULL ) out << ‘,’;

}

else {//子树结点

traverse ( ptr->firstChild ); //向子树方向搜索

if ( ptr->nextSibling != NULL ) out << ‘,’;

}

traverse ( ptr->nextSibling ); //向同一层下一兄弟搜索}

else out << ‘)’;

}

(5) 析构函数清除一棵用广义表表示的树

GenTree :: ~ GenTree ( ) {

//用广义表表示的树的析构函数, 假定first ≠NULL

Remove ( first );

}

void GenTree :: Remove ( GenTreeNode *ptr ) {

GenTreeNode * p;

while ( ptr != NULL ) {

p = ptr->nextSibling;

if ( p->utype == 2 ) Remove ( p->firstChild ); //在子树中删除

ptr ->nextSibling = p ->nextSibling ; delete ( p );

//释放结点p

}

}

6-2 列出右图所示二叉树的叶结点、分支结点和每个结点的层次。 【解答】

二叉树的叶结点有⑥、⑧、⑨。分支结点有①、②、③、④、⑤、⑦。

结点①的层次为0;结点②、③的层次为1;结点④、⑤、⑥的层次

为2;结点⑦、⑧的层次为3;结点⑨的层次为4。

6-3 使用 (1) 顺序表示和 (2) 二叉链表表示法,分别画出右图所示二叉树的存储表示。 【解答】

6-4 用嵌套类写出用链表表示的二叉树的类声明。 【解答】

#include

template class BinaryTree { private:

template class BinTreeNode { public: BinTreeNode *leftChild , *rightChild ; Type data ;

}

Type RefValue ;

BinTreeNode * root ;

BinTreeNode *Parent ( BinTreeNode *start, BinTreeNode *current ); int Insert ( BinTreeNode *current, const Type& item );

int Find ( BinTreeNode *current, const Type& item ) const; void destroy ( BinTreeNode *current )

;

void Traverse ( BinTreeNode *current, ostream& out ) const; public:

BinaryTree ( ) : root ( NULL ) { }

BinaryTree ( Type value ) : RefValue ( value ), root ( NULL ){ }

~BinaryTree ( ) { destroy (root); }

BinTreeNode ( ) : leftChild ( NULL ), rightChild ( NULL ) { }

BinTreeNode ( Type item ) : data ( item ), leftChild ( NULL ), rightChild ( NULL ) { }

顺序表示

Type& GetData ( ) const { return data ; }

BinTreeNode* GetLeft ( ) const { return leftChild ; } BinTreeNode* GetRight ( ) const { return rightChild ; } void SetData ( const Type& item ){ data = item ; } void SetLeft ( BinTreeNode *L ) { leftChild = L ; }

void SetRight ( BinTreeNode *R ){ RightChild =R ; } int IsEmpty ( ) { return root == NULL ? 1 : 0; } BinTreeNode *Parent ( BinTreeNode *current )

{ return root == NULL || root == current ? NULL : Parent ( root, current ); }

BinTreeNode * LeftChild ( BinTreeNode *current )

{ return current != NULL ? current ->leftChild : NULL ; }

BinTreeNode * RighttChild ( BinTreeNode *current )

{ return current != NULL ? current ->rightChild : NULL ; }

int Insert ( const Type& item );

BinTreeNode * Find ( const Type& item ); BinTreeNode * GetRoot ( ) const { return root ; }

friend istream& operator >> ( istream& in, BinaryTree& Tree );

//输入二叉树

friend ostream& operator << ( ostream& out, BinaryTree& Tree ); //输出二叉树

}

6-5 在结点个数为n (n>1)的各棵树中,高度最小的树的高度是多少?它有多少个叶结点?多少个分支结点?高度最大的树的高度是多少?它有多少个叶结点?多少个分支结点? 【解答】

结点个数为n 时,高度最小的树的高度为1,有2层;它有n -1个叶结点,1个分支结点;高度最大的树的高度为n -1,有n 层;它有1个叶结点,n -1个分支结点。

6-6 试分别画出具有3个结点的树和3个结点的二叉树的所有不同形态。 【解答】 具有3个结点的树 具有3个结点的二叉树

6-7 如果一棵树有n 1个度为1的结点, 有n 2个度为2的结点, … , n m 个度为m 的结点, 试问有多少个度为0的结点? 试推导之。 【解答】

总结点数 n = n 0 + n 1 + n 2 + … + n m 总分支数 e = n -1 = n 0 + n 1 + n 2 + … + n m -1

= m*n m + (m -1)*n m -1 + … + 2*n 2 + n 1 则有 1)1(20+??

?

??-=∑=m i i n i n

6-8 试分别找出满足以下条件的所有二叉树:

(1) 二叉树的前序序列与中序序列相同; (2) 二叉树的中序序列与后序序列相同; (3) 二叉树的前序序列与后序序列相同。 【解答】 (1) 二叉树的前序序列与中序序列相同:空树或缺左子树的单支树; (2) 二叉树的中序序列与后序序列相同:空树或缺右子树的单支树;

(3) 二叉树的前序序列与后序序列相同:空树或只有根结点的二叉树。

6-9 若用二叉链表作为二叉树的存储表示,试针对以下问题编写递归算法: (1) 统计二叉树中叶结点的个数。

(2) 以二叉树为参数,交换每个结点的左子女和右子女。 【解答】

(1) 统计二叉树中叶结点个数

int BinaryTree :: leaf ( BinTreeNode * ptr ) { if ( ptr == NULL ) return 0;

else if ( ptr ->leftChild == NULL && ptr ->rightChild == NULL ) return 1; else return leaf ( ptr ->leftChild ) + leaf ( ptr ->rightChild );

}

(2) 交换每个结点的左子女和右子女

void BinaryTree :: exchange ( BinTreeNode * ptr ) { BinTreeNode * temp ;

if ( ptr ->leftChild != NULL || ptr ->rightChild != NULL ) {

temp = ptr ->leftChild ;

ptr ->leftChild = ptr ->rightChild ;

ptr ->rightChild = temp ; exchange ( ptr ->leftChild ); exchange ( ptr ->rightChild );

} }

6-10 一棵高度为h 的满k 叉树有如下性质: 第h 层上的结点都是叶结点, 其余各层上每个结点都有k 棵非空子树, 如果按层次自顶向下, 同一层自左向右, 顺序从1开始对全部结点进行编号, 试问: (1) 各层的结点个数是多少?

(2) 编号为i 的结点的父结点(若存在)的编号是多少?

(3) 编号为i 的结点的第m 个孩子结点(若存在)的编号是多少?

(4) 编号为i 的结点有右兄弟的条件是什么? 其右兄弟结点的编号是多少? (5) 若结点个数为 n, 则高度h 是n 的什么函数关系? 【解答】

(1) k i ( i = 0, 1, ……, h )

(2) ??

????

-+k 2k i

(3) ( i -1)*k + m + 1

(4) ( i -1 ) % k ≠ 0 或 k *k 2k i i ??

????-+≤ 时有右兄弟,右兄弟为i + 1。 (5) h = log k (n*(k -1)+1)-1 (n = 0时h = -1 )

6-11 试用判定树的方法给出在中序线索化二叉树上: 【解答】

(1) 搜索指定结点的在中序下的后继。 设指针q 指示中序线索化二叉树中的指定结点,指针p 指示其后继结点。

找q 的右子树中在中序下的第一个结点的程序为:

p = q ->rightChild ;

while ( p ->leftThread == 1 ) p = p ->leftChild ; // p 即为q 的后继

(2) 搜索指定结点的在前序下的后继。

(3) 搜索指定结点的在后序下的后继。

可用一段遍历程序来实现寻找p 的右子树中后序下的第一个结点:即该子树第一个找到的叶结点。

找到后立即返回它的地址。

6-12 已知一棵完全二叉树存放于一个一维数组T[n]中,T[n]中存放的是各结点的值。试设计一个算法,从T[0]开始顺序读出各结点的值,建立该二叉树的二叉链表表示。 【解答】

template

//建立二叉树

istream& operator >> ( istream& in, BinaryTree& t ) { int n ;

cout << "Please enter the number of node : "; in >> n ;

q ->rightThread == 1?

=

q ->rightChild == NULL ? = q 无后继

≠ q 的后继为->rightChild

q 的后继为q 的右子树中

中序下的第一个结点

q ->leftThread == 0 ?

=

q 的后继为q ->leftChild

q ->rightThread == 0 ?

=

q 的后继为q ->rightChild

p = q ;

while ( p ->rightThread == 1 &&

p ->rightChild != NULL ) p = p ->rightChild ; if ( p ->rightChild ==NULL ) q 无后继; else 的后继为p ->rightChild

( p = parent (q ) ) == NULL ? = q 的后继为p

p -> 1 || p ->rightChild == q ? = ≠ q 的后继为p 的右子树中后序下的第一个结点

q 无后继

Type *A = new Type[n+1];

for ( int i = 0; i < n; i++ ) in >> A[i];

t. ConstructTree( T, n, 0, ptr ); //以数组建立一棵二叉树

delete [ ] A;

return in;

}

template

void BinaryTree :: ConstructTree ( Type T[ ], int n, int i, BinTreeNode *& ptr ) {

//私有函数:将用T[n]顺序存储的完全二叉树, 以i为根的子树转换成为用二叉链表表示的

//以ptr为根的完全二叉树。利用引用型参数ptr将形参的值带回实参。

if ( i >= n )ptr = NULL;

else {

ptr = new BinTreeNode ( T[i] );//建立根结点

ConstructTree ( T, n, 2*i+1, ptr->leftChild );

ConstructTree ( T, n, 2*i+2, ptr->rightChild );

}

}

6-13 试编写一个算法,把一个新结点l作为结点s的左子女插入到一棵中序线索化二叉树中,s原来的左子女变成l的左子女。

【解答】

template

void ThreadTree :: leftInsert ( ThreadNode *s, ThreadNode * l ) {

if ( s != NULL && l != NULL ) {

l->leftChild = s->leftChild;l->leftThread = s->leftThread; //预先链接

l->rightChild = s;l->rightThread = 1; //新插入结点*l的后继为*s

s->leftChild = l;s->leftThread = 0; //*l成为*s的左子女

if ( l->leftThread == 0 ) {//*l的左子女存在

ThreadNode * p = l->leftChild;

while ( p->rightThread == 0 ) //找*l的左子树中序下最后一个结点

p = p->rightChild;

p->rightChild = l;//该结点的后继为*l

}

}

}

6-14 写出向堆中加入数据4, 2, 5, 8, 3, 6, 10, 14时,每加入一个数据后堆的变化。

【解答】以最小堆为例:

6-16 请画出右图所示的树所对应的二叉树。 【解答】

6-17 在森林的二叉树表示中,用llink 存储指向结点第一个子女的指针,用rlink 存储指向结点下一个兄弟的指针,用data 存储结点的值。如果我们采用静态二叉链表作为森林的存储表示,同时按森林的先根次序依次安放森林的所有结点,则可以在它们的结点中用只有一个二进位的标志ltag

代替llink ,用rtag 代替rlink 。并设定若ltag = 0,则该结点没有子女,若ltag ≠ 0,则该结点有子女;若rtag = 0,则该结点没有下一个兄弟,若rtag ≠ 0,则该结点有下一个兄弟。试给出这种表示的结构定义,并设计一个算法,将用这种表示存储的森林转换成用llink - rlink 表示的森林。 【解答】

(1) 结构定义

template class

LchRsibNode { //左子女-右兄弟链表结点类的定义

protected: Type data ;

//结点数据

int llink, rlink ;

//结点的左子女、右兄弟指针

对应二叉树

对应二叉树

llink data

rlink

0 1 2 3 4 5 6 7 8 9 10 森林的左子女-右兄弟

表示的静态二叉链表

ltag data rtag

0 1 2 3 4 5 6 7 8 9 10

森林的双标记表示

public:

LchRsibNode ( ) : llink(NULL), rlink(NULL) { }

LchRsibNode ( Type x ) : llink(NULL), rlink(NULL), data(x) { }

}

template class DoublyTagNode {//双标记表结点类的定义protected:

Type data;//结点数据

int ltag, rtag;//结点的左子女、右兄弟标记public:

DoublyTagNode ( ) : ltag(0), rtag(0) { }

DoublyTagNode ( Type x ) : ltag(0), rtag(0), data(x) { }

}

template class staticlinkList //静态链表类定义

: public LchRsibNode, public DoublyTagNode {

private:

LchRsibNode *V;//存储左子女-右兄弟链表的向量DoublyTagNode *U;//存储双标记表的向量

int MaxSize, CurrentSize;//向量中最大元素个数和当前元素个数public:

dstaticlinkList ( int Maxsz ) : MaxSize ( Maxsz ), CurrentSize (0) {

V = new LchRsibNode [Maxsz];

U = new DoublyTagNode [Maxsz];

}

(2)森林的双标记先根次序表示向左子女-右兄弟链表先根次序表示的转换void staticlinkList :: DtagF-LchRsibF ( ) {

Stack st; int k;

for ( int i = 0;i < CurrentSize;i++ ) {

switch ( U[i].ltag ) {

case 0 : switch ( U[i].rtag ) {

case 0 : V[i].llink = V[i].rlink = -1;

if ( st.IsEmpty ( ) == 0 )

{ k = st.GetTop ( ); st.Pop ( ); V[k].rlink = i + 1; }

break;

case 1 : V[i].llink = -1; V[i].rlink = i + 1; break;

}

break;

case 1 : switch ( U[i].rtag ) {

case 0 : V[i].llink = i + 1; V[i].rlink = -1; break;

case 1 : V[i].llink = i + 1; st.Push ( i );

}

}

}

}

6-18 二叉树的双序遍历(Double -order traversal)是指:对于二叉树的每一个结点来说,先访问这个结点,再按双序遍历它的左子树,然后再一次访问这个结点,接下来按双序遍历它的右子树。试写出执行这种双序遍历的算法。 【解答】

template

void BinaryTree :: Double_order ( BinTreeNode *current ){ if ( current != NULL ) { cout << current ->data << ' '; Double_order ( current ->leftChild ); cout << current ->data << ' '; Double_order ( current ->rightChild );

} }

6-19 已知一棵二叉树的前序遍历的结果是ABECDFGHIJ, 中序遍历的结果是EBCDAFHIGJ, 试画出这棵二叉树。 【解答】

当前序序列为ABECDFGHIJ ,中序序列为EBCDAFHIGJ 时,逐步形成二叉树的过程如下图所示:

6-20 已知一棵树的先根次序遍历的结果与其对应二叉树表示(

长子-兄弟表示)的前序遍历结果相同, 树的后根次序遍历结果与其对应二叉树表示的中序遍历结果相同。试问利用树的先根次序遍历结果和后根次序遍历结果能否唯一确定一棵树? 举例说明。 【解答】

因为给出二叉树的前序遍历序列和中序遍历序列能够唯一地确定这棵二叉树,因此,根据题目给出的条件,利用树的先根次序遍历结果和后根次序遍历结果能够唯一地确定一棵树。 例如,对于题6-16所示的树

对应二叉树的前序序列为1, 2, 3, 4, 5, 6, 8, 7;中序序列为3, 4, 8, 6, 7, 5, 2, 1。

原树的先根遍历序列为 1, 2, 3, 4, 5, 6, 8, 7;后根遍历序列为3, 4, 8, 6, 7, 5, 2, 1。

6-21 给定权值集合{15, 03, 14, 02, 06, 09, 16, 17}, 构造相应的霍夫曼树, 并计算它的带权外部路径长度。 【解答】

此树的带权路径长度WPL = 229。

6-22 假定用于通信的电文仅由8个字母c1, c2, c3, c4, c5, c6, c7, c8组成, 各字母在电文中出现的频率分别为5, 25, 3, 6, 10, 11, 36, 4。试为这8个字母设计不等长Huffman 编码, 并给出该电文的总码数。 【解答】已知字母集 { c1, c2, c3, c4, c5, c6, c7, c8 },频率 {5, 25, 3, 6, 10, 11, 36, 4 },则Huffman 编码为

电文总码数为 4 * 5 + 2 * 25 + 4 * 3 + 4 * 6 + 3 * 10 + 3 * 11 + 2 * 36 + 4 * 4 = 257

6-

23 给定一组权值: 23, 15, 66, 07, 11, 45, 33, 52, 39, 26, 58,

试构造一棵具有最小带权外部路径长度的扩充4叉树, 要求该4叉树中所有内部结点的度都是4, 所有外部结点的度都是0。这棵扩充4

叉树的带

F :

(Ⅰ)

(Ⅱ) (Ⅲ)

(Ⅳ)

(

(Ⅵ)

(Ⅶ)

C3 C8 C1 C4

权外部路径长度是多少?

【解答】权值个数n = 11,扩充4 叉树的内结点的度都为4,而外结点的度都为0。设内结点个数为n 4,外结点个数为n 0,则可证明有关系n 0 = 3 * n 4 + 1。由于在本题中n 0 = 11≠3 * n 4 +1,需要补2个权值为0的外结点。此时内结点个数n 4 = 4。

仿照霍夫曼树的构造方法来构造扩充4叉树,每次合并4个结点。

此树的带权路径长度WPL = 375 + 82 + 169 + 18 = 644。

第六章树和二叉树试题 一、单项选择题 1.树中所有结点的度等于所有结点数加()。 A. 0 B. 1 C. -1 D. 2 2.在一棵树中,()没有前驱结点。 A. 分支结点 B. 叶结点 C. 根结点 D. 空结点 3.在一棵二叉树的二叉链表中,空指针域数等于非空指针域数加()。 A. 2 B. 1 C. 0 D. -1 4.在一棵具有n个结点的二叉树中,所有结点的空子树个数等于()。 A. n B. n-1 C. n+1 D. 2*n 5.在一棵具有n个结点的二叉树的第i层上(假定根结点为第0层,i大于等

于0而小于等于树的高度),最多具有()个结点。 A. 2i B. 2i+1 C. 2i-1 D. 2n 6.在一棵高度为h(假定根结点的层号为0)的完全二叉树中,所含结点个数不 小于()。 A. 2h-1 B. 2h+1 C. 2h-1 D. 2h 7.在一棵具有35个结点的完全二叉树中,该树的高度为()。假定空树 的高度为-1。 A. 5 B. 6 C. 7 D. 8 8.在一棵具有n个结点的完全二叉树中,分支结点的最大编号为()。假 定树根结点的编号为0。 A. ?(n-1)/2? B. ?n/2? C. ?n/2? D. ?n/2? -1 9.在一棵完全二叉树中,若编号为i的结点存在左孩子,则左子女结点的编号 为()。假定根结点的编号为0

A. 2i B. 2i-1 C. 2i+1 D. 2i+2 10.在一棵完全二叉树中,假定根结点的编号为0,则对于编号为i(i>0)的结 点,其双亲结点的编号为()。 A. ?(i+1)/2? B. ?(i-1)/2? C. ?i/2? D. ?i/2? -1 11.在一棵树的左子女-右兄弟表示法中,一个结点的右孩子是该结点的() 结点。 A. 兄弟 B. 子女 C. 祖先 D. 子孙 12.在一棵树的静态双亲表示中,每个存储结点包含()个域。 A. 1 B. 2 C. 3 D. 4 13.已知一棵二叉树的广义表表示为a (b (c), d (e ( , g (h) ), f ) ),则 该二叉树的高度为()。假定根结点的高度为0。 A. 3 B. 4 C. 5 D. 6

习题六树和二叉树 一、单项选择题 1.以下说法错误的是 ( ) A.树形结构的特点是一个结点可以有多个直接前趋 B.线性结构中的一个结点至多只有一个直接后继 C.树形结构可以表达(组织)更复杂的数据 D.树(及一切树形结构)是一种"分支层次"结构 E.任何只含一个结点的集合是一棵树 2.下列说法中正确的是 ( ) A.任何一棵二叉树中至少有一个结点的度为2 B.任何一棵二叉树中每个结点的度都为2 C.任何一棵二叉树中的度肯定等于2 D.任何一棵二叉树中的度可以小于2 3.讨论树、森林和二叉树的关系,目的是为了() A.借助二叉树上的运算方法去实现对树的一些运算 B.将树、森林按二叉树的存储方式进行存储 C.将树、森林转换成二叉树 D.体现一种技巧,没有什么实际意义 4.树最适合用来表示 ( ) A.有序数据元素 B.无序数据元素 C.元素之间具有分支层次关系的数据 D.元素之间无联系的数据 5.若一棵二叉树具有10个度为2的结点,5个度为1的结点,则度为0的结点个数是()A.9 B.11 C.15 D.不确定 6.设森林F中有三棵树,第一,第二,第三棵树的结点个数分别为M1,M2和M3。与森林F对应的二叉树根结点的右子树上的结点个数是()。 A.M1 B.M1+M2 C.M3 D.M2+M3 7.一棵完全二叉树上有1001个结点,其中叶子结点的个数是() A. 250 B. 500 C.254 D.505 E.以上答案都不对 8. 设给定权值总数有n 个,其哈夫曼树的结点总数为( ) A.不确定 B.2n C.2n+1 D.2n-1 9.二叉树的第I层上最多含有结点数为() A.2I B. 2I-1-1 C. 2I-1 D.2I -1 10.一棵二叉树高度为h,所有结点的度或为0,或为2,则这棵二叉树最少有( )结点A.2h B.2h-1 C.2h+1 D.h+1 11. 利用二叉链表存储树,则根结点的右指针是()。 A.指向最左孩子 B.指向最右孩子 C.空 D.非空 14.在二叉树结点的先序序列,中序序列和后序序列中,所有叶子结点的先后顺序()A.都不相同 B.完全相同 C.先序和中序相同,而与后序不同 D.中序和后序相同,而与先序不同 15.在完全二叉树中,若一个结点是叶结点,则它没()。 A.左子结点 B.右子结点 C.左子结点和右子结点 D.左子结点,右子结点和兄弟结点 16.在下列情况中,可称为二叉树的是()

第六章 树和二叉树 一、选择题 1.已知一算术表达式的中缀形式为 A+B*C-D/E ,后缀形式为ABC*+DE/-,其前缀形式为 ( ) A .-A+B*C/DE B. -A+B*CD/E C .-+*ABC/DE D. -+A*BC/DE 【北京航空航天大学 1999 一、3 (2分)】 2.算术表达式a+b*(c+d/e )转为后缀表达式后为( )【中山大学 1999 一、5】 A .ab+cde/* B .abcde/+*+ C .abcde/*++ D .3. 设有一表示算术表达式的二叉树(见下图), 它所表示的算术表达式是( ) 【南京理工大学1999 一、20(2分)】 A. A*B+C/(D*E)+(F-G) B. (A*B+C)/(D*E)+(F-G) C. (A*B+C)/(D*E+(F-G )) D. A*B+C/D*E+F-G 4. 设树T 的度为4,其中度为1,2,3和4的结点个数分别为4,2,1 ,1 则T 中的叶子数为( ) A .5 B .6 C .7 D .8 【南京理工大学 2000 一、8 (1.5分)】 5. 在下述结论中,正确的是( )【南京理工大学 1999 一、4 (1分)】 ①只有一个结点的二叉树的度为0; ②二叉树的度为2; ③二叉树的左右子树可任意 交换; ④深度为K 的完全二叉树的结点个数小于或等于深度相同的满二叉树。 A .①②③ B .②③④ C .②④ D .①④ 6. 设森林F 对应的二叉树为B ,它有m 个结点,B 的根为p,p 的右子树结点个数为n,森林F 中第一棵树的结点个数是( ) A .m-n B .m-n-1 C .n+1 D .条件不足,无法确定 【南京理工大学2000 一、 17(1.5分)】 7. 树是结点的有限集合,它( (1))根结点,记为T 。其余结点分成为m (m>0)个((2)) 的集合T1,T2, …,Tm ,每个集合又都是树,此时结点T 称为Ti 的父结点,Ti 称为T 的子结点(1≤i ≤m )。一个结点的子结点个数称为该结点的( (3) )。二叉树与树是两个 不同的概念,二叉树也是结点的有限集合,它((4))根结点。可以把树的根结点的层数定 义为1,其他结点的层数等于其父结点所在层数加上1。令T 是一棵二叉树,Ki 和Kj 是T 中子结点数小于2的结点中的任意两个,它们所在的层数分别为λKi 和λKj ,当关系式│ λKi-λKj │≤1一定成立时,则称T 为一棵((5))。供选择的答案: (1)(4) A. 有0个或1个 B. 有0个或多个 C. 有且只有一个 D. 有1个或1 个以上 (2) A. 互不相交 B.允许相交 C.允许叶结点相交 D.允许树枝结点相交 (3) A. 权 B.维数 C.次数 D.序 (5) A. 丰满树 B.查找树 C.平衡树 D.完全树 【上海海运学院1999二、 2(5分)】 8.若一棵二叉树具有10个度为2的结点,5个度为1的结点,则度为0的结点个数是( ) A .9 B .11 C .15 D .不确定 【北京工商大学2001一.7(3 分)】 9.在一棵三元树中度为3的结点数为2个,度为2的结点数为1个,度为1的结点数为2

一、选择题 1、设T是一棵树,T’是对应于x的二叉树,则T的先根次序遍历和T’的()次序遍历相同。 A、先根 B、中根 C、后根 D、以上都不是 2、 3、若二叉树的后序遍历序列为dabec,中序遍历序列为debac,则前序序列遍历为()。 A、acbed B、decab C、deabc D、cedba 4、具有35个结点的完全二叉树的深度为() A、5 B、6 C、7 D、8 5、将一棵有100个结点的完全二叉树从上到下,从左到右依次对结点进行编号,根结点的编号为1,则编号为49的结点的左孩子结点编号为() A、98 B、99 C、50 D、48 6、某二叉树的前序和后序序列正好相反,则该二叉树一定是()的二叉树。 A、空或只有一个结点 B、高度等于其结点数 C、任一结点无左孩子 D、任一结点无左孩子 7、二叉树在线索化后,仍不能有效求解的问题是() A、先根线索二叉树中求先根后继 B、中根线索二叉树中求中根后继 C、中根线索二叉树中求中根前驱 D、后根线索二叉树中求后根后继 8、在线索化二叉树中,t所指结点没有左子树的充足条件是() A、t-lchild==NULL B、t->ltag==1 C、t->ltag==1&&t->lchild==NULL D、以上都不对 9、设高度为h的二叉树上只有度为0和度为2的结点,则此类二叉树中所包含的结点数至少为() A、2h B、2h-1 C、2h+1 D、h+1 10、深度为5的二叉树至多有()个结点。 A、16 B、32 C、31 D、10 11、在一非空二叉树的中序遍历序列中,根结点的右边() A、只有右子树上的所有结点 B、只有右子树上的部分结点 C、只有左子树上的所有结点 D、只有左子树上的部分结点 12、树最适合用来表示() A、有序数据元素 B、无序数据元素 C、元素之间具有分支层次关系的数据 D、元素之间无联系的数据 13、任何一棵二叉树的叶结点在先序、中序和后序遍历序列中的相对次序() A、不发生改变 B、发生改变 C、不能确定 D、以上都不对 14、设n,m为一棵二叉树上的两个结点,在中序遍历时,n在m前的条件是() A、n在m右方 B、n是m祖先 C、n在m左方 D、n是m子孙 15、线索二叉树是一种()结构 A、逻辑 B、逻辑和存储 C、物理 D、线性 16、森林的后根遍历序列与其对应二叉树的()遍历序列一致。 A、先根 B、后根 C、中根 D、不可能 二、填空题 1、由一棵二叉树后序序列和(中序)可唯一确定这棵二叉树。 2、含有n个结点的二叉树用二叉链表表示时,有(N+1)个空链域。 3、有m个叶子结点的哈夫曼树有(2*M-1)个结点。

第六章答案 6. 1分别画出具有3个结点的树和3个结点的二叉树的所有不同形态。 【解答】 具有3个结点的树具有3个结点的二叉树 6.3已知一棵度为k的树中有n1个度为1的结点,n2个度为2的结点,……,n k个度为k 的结点,则该树中有多少个叶子结点? 【解答】设树中结点总数为n,则n=n0 + n1 + …… + n k 树中分支数目为B,则B=n1 + 2n2 + 3n3+ …… + kn k 因为除根结点外,每个结点均对应一个进入它的分支,所以有n= B + 1 即n0 + n1 + …… + n k = n1 + 2n2 + 3n3+ …… + kn k + 1 由上式可得叶子结点数为:n0 = n2 + 2n3+ …… + (k-1)n k + 1 6.5已知二叉树有50个叶子结点,则该二叉树的总结点数至少应有多少个? 【解答】n0表示叶子结点数,n2表示度为2的结点数,则n0 = n2+1 所以n2=n0 –1=49,当二叉树中没有度为1的结点时,总结点数n=n0+n2=99 6.6 试分别找出满足以下条件的所有二叉树: (1) 前序序列与中序序列相同; (2) 中序序列与后序序列相同; (3) 前序序列与后序序列相同。 【解答】 (1) 前序与中序相同:空树或缺左子树的单支树; (2) 中序与后序相同:空树或缺右子树的单支树; (3) 前序与后序相同:空树或只有根结点的二叉树。 6.9 假设通讯的电文仅由8个字母组成,字母在电文中出现的频率分别为: 0.07,0.19,0.02,0.06,0.32,0.03,0.21,0.10 请为这8个字母设计哈夫曼编码。 【解答】 构造哈夫曼树如下:

第六章树和二叉树 一、选择题 1.已知一算术表达式的中缀形式为 A+B*C-D/E,后缀形式为ABC*+DE/-,其前缀形式为( D ) A.-A+B*C/DE B. -A+B*CD/E C.-+*ABC/DE D. -+A*BC/DE 2.算术表达式a+b*(c+d/e)转为后缀表达式后为( B ) A.ab+cde/* B.abcde/+*+ C.abcde/*++ D.abcde*/++ 3.设有一表示算术表达式的二叉树(见下图), 它所表示的算术表达式是( C ) A. A*B+C/(D*E)+(F-G) B. (A*B+C)/(D*E)+(F-G) C. (A*B+C)/(D*E+(F-G)) D. A*B+C/D*E+F-G 4.设树T的度为4,其中度为1,2,3和4的结点个数分别为4,2,1,1 则T中的叶子数为( D )A.5 B.6 C.7 D.8 5.在下述结论中,正确的是( D ) ①只有一个结点的二叉树的度为0; ②二叉树的度为2;③二叉树的左右子树可任意交换; ④深度为K的完全二叉树的结点个数小于或等于深度相同的满二叉树。 A.①②③ B.②③④ C.②④ D.①④ 6.设森林F对应的二叉树为B,它有m个结点,B的根为p,p的右子树结点个数为n,森林F中第一棵树的结点个数是( A ) A.m-n B.m-n-1 C.n+1 D.条件不足,无法确定 7. 树是结点的有限集合,它( C )根结点,记为T。其余结点分成为m(m>0)个(A)的集合T1,T2,…,Tm,每个集合又都是树,此时结点T称为Ti的父结点,Ti称为T的子结点(1≤i≤m)。一个结点的子结点个数称为该结点的(C )。二叉树与树是两个不同的概念,二叉树也是结点的有限集合,它(A)根结点。可以把树的根结点的层数定义为1,其他结点的层数等于其父结点所在层数加上1。令T是一棵二叉树,Ki和Kj是T中子结点数小于2的结点中的任意两个,它们所在的层数分别为λKi和λKj,当关系式│λKi-λKj│≤1一定成立时,则称T为一棵(C)。供选择的答案: (1)(4) A.有0个或1个 B.有0个或多个 C.有且只有一个 D.有1个或1个以上 (2) A.互不相交 B.允许相交 C.允许叶结点相交 D.允许树枝结点相交 (3) A.权 B.维数 C.次数 D.序 (5) A.丰满树 B.查找树 C.平衡树 D.完全树 8.若一棵二叉树具有10个度为2的结点,5个度为1的结点,则度为0的结点个数是(B ) A.9 B.11 C.15 D.不确定 9.在一棵三元树中度为3的结点数为2个,度为2的结点数为1个,度为1的结点数为2个,则度为0的结点数为( C )个 A.4 B.5 C.6 D.7 10.设森林F中有三棵树,第一,第二,第三棵树的结点个数分别为M1,M2和M3。与森林F对应的二叉树根结点的右子树上的结点个数是( D )。 A.M1 B.M1+M2 C.M3 D.M2+M3 11.具有10个叶结点的二叉树中有( B)个度为2的结点, A.8 B.9 C.10 D.ll 12.一棵完全二叉树上有1001个结点,其中叶子结点的个数是( E ) A. 250 B.500 C.254 D.505 E.以上答案都不对 13.设给定权值总数有n个,其哈夫曼树的结点总数为( D ) A.不确定 B.2n C.2n+1 D.2n-1 14.有n个叶子的哈夫曼树的结点总数为( D )。 A.不确定 B.2n C.2n+1 D.2n-1 15.若度为m的哈夫曼树中,其叶结点个数为n,则非叶结点的个数为(C )。

第6章树和二叉树作业 1、假设在树中,结点x是结点y的双亲时,用(x,y)来表示树边。已知一棵树的树边集合为{ (e,i), (b,e), (b,d), (a,b), (g,j), (c,g), (c,f), (h,l), (c,h), (a,c) } ,用树型表示法表示该树,并回答下列问题: ①哪个是根结点? 哪些是叶子结点? 哪个是g的双亲? 哪些是g的祖先? 哪些是g的孩子? 那些是e的子孙? 哪些是e的兄弟? 哪些是f 的兄弟? ②b和n的层次各是多少? 树的深度是多少? 以结点c为根的子树的深度是多少? 2、一棵深度为h的满k叉树有如下性质:第h层上的结点都是叶子结点,其余各层上每个结点都有k棵非空子树。如果按层次顺序(同层自左至右)从1开始对全部结点编号,问: ①各层的结点数是多少? ②编号为i的结点的双亲结点(若存在)的编号是多少? ③编号为i的结点的第j个孩子结点(若存在)的编号是多少? ④编号为i的结点的有右兄弟的条件是什么? 其右兄弟的编号是多少? 3、设有如图6-27所示的二叉树。 ①分别用顺序存储方法和链接存储方法画 出该二叉树的存储结构。 ②写出该二叉树的先序、中序、后序遍历序 列。 4、已知一棵二叉树的先序遍历序列和中序遍 历序列分别为ABDGHCEFI和 GDHBAECIF,请画出这棵二叉树,然后给 出该树的后序遍历序列。

5、设一棵二叉树的中序遍历序列和后序遍历序列分别为BDCEAFHG 和DECBHGFA ,请画出这棵二叉树,然后给出该树的先序序列。 6、已知一棵二叉树的中序遍历序列和后序遍历序列分别为dgbaekchif 和gdbkeihfca,请画出这棵二叉树对应的中序线索树和后序线索树。 7、以二叉链表为存储结构,请分别写出求二叉树的结点总数及叶子结点总数的算法。 8、设图6-27所示的二叉树是森林F所对应的二叉树,请画出森林F。 9、设有一棵树,如图6-28 所示。 ①请分别用双亲表示法、孩 子表示法、孩子兄弟表示法 给出该树的存储结构。 ②请给出该树的先序遍历 序列和后序遍历序列。 ③请将这棵树转换成二叉 树。 10、设给定权值集合 w={3,5,7,8,11,12} ,请构造 关于w的一棵huffman树,并求其加权路径长度WPL 。 11、假设用于通信的电文是由字符集{a, b, c, d, e, f, g, h}中的字符构成,这8个字符在电文中出现的概率分别为{0.07, 0.19, 0.02, 0.06, 0.32, 0.03, 0.21, 0.10} 。 ①请画出对应的huffman树(按左子树根结点的权小于等于右子树根结点的权的次序构造)。 ②求出每个字符的huffman编码。

第六章树和二叉树 一、应用题 1.已知完全二叉树的第七层有10个叶子结点,则整个二叉树的结点数最多是多少? 2.高度为10的二叉树,其结点最多可能为多少? 3.任意一个有n个结点的二叉树,已知它有m个叶子结点,试证明非叶子结点有(m-1)个度为2,其余度为1。 4. 已知A[1..N]是一棵顺序存储的完全二叉树,如何求出A[i]和A[j]的最近的共同祖先? 5.已知一棵满二叉树的结点个数为20到40之间的素数,此二叉树的叶子结点有多少个?

6.一棵共有n个结点的树,其中所有分支结点的度均为K,求该树中叶子结点的个数。 7.设T是具有n个内结点的扩充二叉树,I是它的内路径长度,E是它的外路径长度。试利用归纳法证明E=I+2n, n>=0. 8.试证明:同一棵二叉树的所有叶子结点,在前序序列、中序序列以及后序序列中都按相同的相对位置出现(即先后顺序相同),例如前序abc,后序bca,对称序bac。 9. 由二叉树的中序序列及前序序列能唯一的建立二叉树,试问中序序列及后序序列是否也能唯一的建立二叉树,不能则说明理由,若能对中序序列DBEAFGC 和后序序列DEBGFCA构造二叉树。

10. 由一棵二叉树的前序序列和中序序列可唯一确定这棵二叉树。设一棵二叉树的前序序列为ABDGECFH,中序序列为:DGBEAFHC 。试画出该二叉树。 二、算法设计题 1. 给出算法将二叉树表示的表达式二叉树按中缀表达式输出,并加上相应的括号。 2.编程求以孩子—兄弟表示法存储的森林的叶子结点数。

3.要求二叉树按二叉链表形式存储, (1)写一个建立二叉树的算法。 (2)写一个判别给定的二叉树是否是完全二叉树的算法。 完全二叉树定义为:深度为K,具有N个结点的二叉树的每个结点都与深度为K 的满二叉树中编号从1至N的结点一一对应。此题以此定义为准。 4.假设以双亲表示法作树的存储结构,写出双亲表示的类型说明,并编写求给定的树的深度的算法。(注:已知树中结点数)

第六章课后习题 6、1、各层的结点数目是:K 2、编号为n的结点的双亲结点是:<=(n-2)/k的最大整数 3、编号为n的结点的第i个孩子结点编号是:k*(n-1)+1+i 4、编号为n的结点有右兄弟的条件是:(n-1)能被k整除 右兄弟的编号是:n+1. 7、1、先序序列和中序序列相同:空二叉树或没有左子树的二叉树。 2、中序序列和后序序列相同:空二叉树或没有右子树的二叉树。 3、先序序列和后序序列相同:空二叉树或只有根的二叉树。 9、中序序列:BDCEAFHG和后序序列:DECBHGFA的二叉树为: A B F C G D E H 先序序列:ABCDEFGH 算法设计: 3、typedef struct { int data[100]; int top; }seqstack; seqstack *s; Perorder(char a[],int n) { int i=1,count=1; s->top=-1;

if(n==0)return(0); else { if(I<=n) { s->top++; s->data[s->top]=a[I]; } while(countdata[s->top]); count++; s->top--; if(s->data[s->top]);==a[i]) { printf(“%c”,s->data[s->top]); count++; s->top--; } if((2*i+1)top++; s->data[s->top]=a[i+1]; s->top++; s->data[s->top]=a[i]; } else if(a*itop++; s->data[s->top]=a[i]; } else if(i/2%2==1)i=i/2/2+1; else i=i/2+1; } } } main() { char A[]=“kognwyuvb”; int n=strlen(A); s=(seqstack *)malloc(sizeof(seqstack)); printf(“\n”); Perorder(A,n);

一、判断题 (√)1. 若二叉树用二叉链表作存贮结构,则在n个结点的二叉树链表中只有n—1个非空指针域。 (×)2.二叉树中每个结点的两棵子树的高度差等于1。 (√)3.二叉树中每个结点的两棵子树是有序的。 (×)4.二叉树中每个结点有两棵非空子树或有两棵空子树。 (×)5.二叉树中所有结点个数是2k-1-1,其中k是树的深度。(应2i-1) (×)6.二叉树中所有结点,如果不存在非空左子树,则不存在非空右子树。 (×)7.对于一棵非空二叉树,它的根结点作为第一层,则它的第i层上最多能有2i —1个结点。(应2i-1) (√)8.用二叉链表法存储包含n个结点的二叉树,结点的2n个指针区域中有n+1个为空指针。 (√)9.具有12个结点的完全二叉树有5个度为2的结点。 ( ) 10、哈夫曼树中没有度为1的结点,所以必为满二叉树。 ( )11、在哈夫曼树中,权值最小的结点离根结点最近。 ( )12、线索二叉树是一种逻辑结构。 (√)13、深度为K的完全二叉树至少有2K-1个结点。 (√ )14、具有n个结点的满二叉树,其叶结点的个数为(n+1)/2。 (√ )15、前序和中序遍历用线索树方式存储的二叉树,不必使用栈。 (╳ )16、哈夫曼树是带权路径长度最短的树,路径上权值较大的点离根较远。 (√)17、在二叉树结点的先序序列和后序序列中,所有叶子结点的先后顺序完全相同。(√)18、二叉树的遍历操作实际上是将非线性结构线性化的过程 (√)19、树的先根遍历序列与其所转化的二叉树的先序遍历序列相同。 (╳)20、树的后根遍历序列与其所转化的二叉树的后序遍历序列相同。 二、填空 1.由3个结点所构成的二叉树有 5 种形态。 2. 线索二叉树的左线索指向其_前驱_____,右线索指向其__后继____。 3.一棵具有257个结点的完全二叉树,它的深度为 9 。 4、如某二叉树有20个叶子结点,有30个结点仅有一个孩子,则该二叉树的总结点数 为_69_____。 5. 设一棵完全二叉树具有1000个结点,则此完全二叉树有 500 个叶子结点,有 499 个度为2的结点,有 1 个结点只有非空左子树,有 0 个结点只有非空右子树。答:最快方法:用叶子数=[n/2]=500 ,n2=n0-1=499。另外,最后一结点为2i属于 左叶子,右叶子是空的,所以有1个非空左子树。完全二叉树的特点决定不可能有左空 右不空的情况,所以非空右子树数=0. 6. 一棵含有n个结点的k叉树,可能达到的最大深度为n,最小深度为 2 。

一、判断题 (√)1.若二叉树用二叉链表作存贮结构,则在n个结点的二叉树链表中只有n—1个非空指针域。(×)2.二叉树中每个结点的两棵子树的高度差等于1。 (√)3.二叉树中每个结点的两棵子树是有序的。 (×)4.二叉树中每个结点有两棵非空子树或有两棵空子树。 (×)5.二叉树中所有结点个数是2k-1-1,其中k是树的深度。(应2i-1) (×)6.二叉树中所有结点,如果不存在非空左子树,则不存在非空右子树。 (×)7.对于一棵非空二叉树,它的根结点作为第一层,则它的第i层上最多能有2i—1个结点。(应2i-1)(√) (√) ( )10 ( )11 ( )12 (√) (√)14 (√)15 (╳)16 (√) (√) (√) (╳) 1 2. 3 4 5.1 答:20 所以有1个非空左子树。完全二叉树的特点决定不可能有左空右不空的情况,所以非空右子树数=0. 6.一棵含有n个结点的k叉树,可能达到的最大深度为n,最小深度为2。 7.若已知一棵二叉树的前序序列是BEFCGDH,中序序列是FEBGCHD,则它的后序序列必是FEGHDCB。 8.在二叉树中,指针p所指结点为叶子结点的条件是_p->lchild==null&&p->rchlid==null?。 三、选择题 1.某二叉树结点的中序序列为A、B、C、D、E、F、G,后序序列为B、D、C、A、F、G、E,则其左子树中结点数目为(C)

A)3 B)2 C)4D)5 2.二叉树是非线性数据结构,所以(C)。 A、它不能用顺序存储结构存储;B、它不能用链式存储结构存储; C、顺序存储结构和链式存储结构都能存储;D、顺序存储结构和链式存储结构都不能使用 3.具有n(n>0)个结点的完全二叉树的深度为(C)。 (A)?log2(n)?(B)?log2(n)?(C)?log2(n)?+1(D)?log2(n)+1? 4.把一棵树转换为二叉树后,这棵二叉树的形态是(A)。 (A)唯一的(B)有多种 5. A 6 号为1 A、 7 A) 8、1, A、 9 A C 10 A 11.? A.2k 12 A. 13.有关二叉树下列说法正确的是(???B) A.二叉树的度为2???B.一棵二叉树的度可以小于2 C.二叉树中至少有一个结点的度为2??D.二叉树中任何一个结点的度都为2 14.一个具有1025个结点的二叉树的高h为(???C) A.11????B.10?????C.11至1025之间?????D.10至1024之间 15.一棵二叉树高度为h,所有结点的度或为0,或为2,则这棵二叉树最少有(B???)结点A.2h????B.2h-1???????C.2h+1????????D.h+1?? 16.对于有n个结点的二叉树,其高度为(D???)

一、填空题 1. 不相交的树的聚集称之为森林。 2. 从概念上讲,树与二叉树是两种不同的数据结构,将树转化为二叉树的基本目的是_树可采用孩子-兄弟链表(二叉链表)做存储结构,目的是利用二叉树的已有算法解决树的有关问题。 3. 深度为k的完全二叉树至少有2 k-1个结点。至多有2 k-1个结点,若按自上而下,从左到右次序给结点编号(从1开始),则编号最小的叶子结点的编号是2 k-2+1。 4. 在一棵二叉树中,度为零的结点的个数为n ,度为2的结点的个 数为n 2,则有n = n 2 +1。 5. 一棵二叉树的第i(i≥1)层最多有2 i-1个结点;一棵有n(n>0)个结点的满二叉树共有(n+1)/2个叶子和(n-1) /2个非终端结点。 6.现有按中序遍历二叉树的结果为abc,问有5种不同形态的二叉树 可以得到这一遍历结果。 7. 哈夫曼树是带权路径长度最小的二叉树。 8. 前缀编码是指任一个字符的编码都不是另一个字符编码的前缀的一种编码方法,是设计不等长编码的前提。 9. 以给定的数据集合{4,5,6,7,10,12,18}为结点权值构造的Huffman树的加权路径长度是 165 。 10. 树被定义为连通而不具有回路的(无向)图。 11. 若一棵根树的每个结点最多只有两个孩子,且孩子又有左、右之分,次序不能颠倒,则称此根树为二叉树。 12. 高度为k,且有个结点的二叉树称为二叉树。 2k-1 满 13. 带权路径长度最小的二叉树称为最优二叉树,它又被称为树。 Huffman 14. 在一棵根树中,树根是为零的结点,而为零的结点是结点。 入度出度树叶

15. Huffman树中,结点的带权路径长度是指由到之间的路径长度与结点权值的乘积。 结点树根 16. 满二叉树是指高度为k,且有个结点的二叉树。二叉树的每一层i上,最多有个结点。 2k-1 2i-1 二、单选题 1. 具有10个叶结点的二叉树中有 (B) 个度为2的结点。 (A)8 (B)9 (C)10 (D)11 2.对二叉树的结点从1开始进行连续编号,要求每个结点的编号大于其左右孩子的编号,同一结点的左右孩子中,其左孩子的编号小于其右孩子的编号,则可采用_(3)次序的遍历实现编号。 (1)先序(2)中序 (3)后序(4)从根开始按层遍历 3. 由2、3、4、7作为结点权值构造的哈夫曼树树的加权路径长度 B 。 A、33 B、30 C、36 D、40 4. 高度为6的满二叉树,总共有的结点数是 B 。 A、15 B、63 C、20 D、25 5. 下面描述根树转换成二叉树的特性中,正确的是 C 。 A、根树转换成的二叉树是唯一的,二叉树的根结点有左、右孩子。 B、根树转换成的二叉树是不唯一的,二叉树的根结点只有左孩子。 C、根树转换成的二叉树是唯一的,二叉树的根结点只有左孩子。 D、根树转换成的二叉树是不唯一的,二叉树的根结点有左、右孩子。 6. 如图所示的4棵二叉树中,不是完全二叉树的是。 A、○ B、○ ○○○○ ○○○○○○

第 6 章 树和二叉树
一、选择题
1.D 2.B 3.C 4.D 5.D 6.A 7.1C 7.2A 7.3C 7.4A 7.5 8.B
C
9.C 10.D 11.B 12.E 13.D 14.D 15.C 16.B 17.C 18.C 19. 20.
BD
21.A 22.A 23.C 24.C 25.C 26.C 27.C 28.C 29.B 30.C 31. 32.
DB
33.A 34.D 35.B 36.B 37.C 38.B 39.B 40.B 41.1 41.2 42. 43.
F
B
CB
44.C 45.C 46.B 47.D 48.B 49.C 50.A 51.C 52.C 53.C 54. 55.
DC
56.B 57.A 58.D 59.D 60.B 61.1 61.2 61.3 62.B 63.B 64. 65.
B
A
G
DD
66.1 66.2 66.3 66.4 66.5
C
D
F
H
I
部分答案解释如下。
12. 由二 叉树 结点的 公式 : n=n0+n1+n2=n0+n1+(n0-1)=2n0+n1-1 , 因 为 n=1001, 所以
1002=2n0+n1,在完全二叉树树中,n1 只能取 0 或 1,在本题中只能取 0,故 n=501,因此选
E。
42.前序序列是“根左右”,后序序列是“左右根”,若要这两个序列相反,只有单支树,所
以本题的 A 和 B 均对,单支树的特点是只有一个叶子结点,故 C 是最合适的,选 C。A 或 B
都不全。由本题可解答 44 题。
47. 左子树为空的二叉树的根结点的左线索为空(无前驱),先序序列的最后结点的右线索
为空(无后继),共 2 个空链域。
52.线索二叉树是利用二叉树的空链域加上线索,n 个结点的二叉树有 n+1 个空链域。
二、判断题
1.× 2.× 3.× 4. 5. √ 6. 7.√ 8.× 9. 10. 11. 12.


√×××
13. 14. 15. 16. 17.√ 18. 19. 20. 21. 22. 23. 24.
×√××
√×√×√××
25. 26. 27. 28. 29.√ 30. 31. 32. 33. 34. 35. 36.
√×××
××√×××√
37. 38. 39. 40. 41.(3) 42. 43. 44. 45. 46. 47. 48.
√×××
√√×√×××
49. 50.
√√
部分答案解释如下。
6.只有在确定何序(前序、中序、后序或层次)遍历后,遍历结果才唯一。
19.任何结点至多只有左子树的二叉树的遍历就不需要栈。
24. 只对完全二叉树适用,编号为 i 的结点的左儿子的编号为 2i(2i<=n),右儿子是 2i+1
(2i+1<=n)
37. 其中序前驱是其左子树上按中序遍历的最右边的结点(叶子或无右子女),该结点无右

第6章树和二叉树(2)

第六章树和二叉树 一、选择题 1.算术表达式a+b*(c+d/e)转为后缀表达式后为() A.ab+cde/* B.abcde/+*+ C.abcde/*++ D.abcde*/++ 2. 设森林F对应的二叉树为B,它有m个结点,B的根为p,p的右子树结点个数为n,森林F中第一棵树的结点个数是() A.m-n B.m-n-1 C.n+1 D.条件不足,无法确定 3.若度为m的哈夫曼树中,其叶结点个数为n,则非叶结点的个数为()。 A.n-1 B.?n/m?-1 C.?(n-1)/(m-1)? D.?n/(m-1)?-1 E.?(n+1)/(m+1)?-1 4.深度为h的满m叉树的第k层有()个结点。(1=

A.(00,01,10,11) B.(0,1,00,11) C.(0,10,110,111)D.(1,01,000,001) 二、判断题 1. 给定一棵树,可以找到唯一的一棵二叉树与之对应。 2.将一棵树转成二叉树,根结点没有左子树; 3. 在中序线索二叉树中,每一非空的线索均指向其祖先结点。 4. 一棵哈夫曼树的带权路径长度等于其中所有分支结点的权值之和。 5.当一棵具有n个叶子结点的二叉树的WPL值为最小时,称其树为Huffman树,且其二叉树的形状必是唯一的。 三、填空题 1.一棵树T中,包括一个度为1的结点,两个度为2的结点,三个度为3的结点,四个度为4的结点和若干叶子结点,则T的叶结点数为___ ___。【山东大学 2001 三、2 (2分)】

一、基础知识题 6.1设树T的度为4,其中度为1,2,3和4的结点个数分别为4,2,1,1,求树T中的叶子数。 【解答】设度为m的树中度为0,1,2,…,m的结点数分别为n0, n1, n2,…, nm,结点总数为n,分枝数为B,则下面二式成立 n= n0+n1+n2+…+nm (1) n=B+1= n1+2n2 +…+mnm+1 (2) 由(1)和(2)得叶子结点数n0=1+ 即: n0=1+(1-1)*4+(2-1)*2+(3-1)*1+(4-1)*1=8 6.2一棵完全二叉树上有1001个结点,求叶子结点的个数。 【解答】因为在任意二叉树中度为2 的结点数n2和叶子结点数n0有如下关系:n2=n0-1,所以设二叉树的结点数为n, 度为1的结点数为n1,则 n= n0+ n1+ n2 n=2n0+n1-1 1002=2n0+n1 由于在完全二叉树中,度为1的结点数n1至多为1,叶子数n0是整数。本题中度为1的结点数n1只能是0,故叶子结点的个数n0为501. 注:解本题时要使用以上公式,不要先判断完全二叉树高10,前9层是满二叉树,第10层都是叶子,……。虽然解法也对,但步骤多且复杂,极易出错。 6.3 一棵124个叶结点的完全二叉树,最多有多少个结点。 【解答】由公式n=2n0+n1-1,当n1为1时,结点数达到最多248个。 6.4.一棵完全二叉树有500个结点,请问该完全二叉树有多少个叶子结点?有多少个度为1的结点?有多少个度为2的结点?如果完全二叉树有501个结点,结果如何?请写出推导过程。

【解答】由公式n=2n0+n1-1,带入具体数得,500=2n0+n1-1,叶子数是整数,度为1的结点数只能为1,故叶子数为250,度为2的结点数是249。 若完全二叉树有501个结点,则叶子数251,度为2的结点数是250,度为1的结点数为0。 6.5 某二叉树有20个叶子结点,有30个结点仅有一个孩子,则该二叉树的总结点数是多少。 【解答】由公式n=2n0+n1-1,得该二叉树的总结点数是69。 6.6 求一棵具有1025个结点的二叉树的高h。 【解答】该二叉树最高为1025(只有一个叶子结点),最低高为11。因为210-1<1025<211-1,故1025个结点的二叉树最低高为11。 6.7 一棵二叉树高度为h,所有结点的度或为0,或为2,则这棵二叉树最少有多少结点。 【解答】第一层只一个根结点,其余各层都两个结点,这棵二叉树最少结点数是2h-1。 6.8将有关二叉树的概念推广到三叉树,则一棵有244个结点的完全三叉树的高度是多少。 【解答】设含n个结点的完全三叉树的高度为h,则有

第六章树和二叉树 int Is_Descendant_C(int u,int v) int Bitree_Sim(Bitree B1,Bitree B2)次根据栈顶元素的mark域值决定做何种动作. typedef struct { int data; EBTNode *lchild; EBTNode *rchild; EBTNode *parent; enum {0,1,2} mark; } EBTNode,EBitree; typedef struct { int data; PBTNode *lchild; PBTNode *rchild; PBTNode *parent; } PBTNode,PBitree; . scanf("%d",&k); c=0; . } void LayerOrder(Bitree T)相比,作了一个修改,不管当前结点是否有左右孩子,都入队列.这样当树为完全二叉树时,遍历时得到是一个连续的不包含空指针的序列.反之,则序列中会含有空指针. Status CreateBitree_Triplet(Bitree &T)意:为了能用一个统一的公式计算子孙数目,所以当T为空指针时,要返回-1而不是0. BTNode *PreOrder_Next(BTNode *p)不是哪儿理解错了 Bitree PostOrder_Next(Bitree p)irstchild) return 1; for(sd=1,p=[T].firstchild;p;p=p->next) if((d=SubDepth(p->child))>sd) sd=d; return sd+1; }arent) dep++; . Build_Sub(1,n,1,n); . } 分析:本算法利用了这样一个性质,即一棵子树在前序和中序序列中所占的位置总是连续的.因此,就可以用起始下标和终止下标来确定一棵子树.Pre_Start,Pre_End,In_Start和

第六章树及二叉树 一、下面是有关二叉树的叙述,请判断正误 (√)1. 若二叉树用二叉链表作存贮结构,则在n个结点的二叉树链表中只有n—1个非空指针域。 (×)2.二叉树中每个结点的两棵子树的高度差等于1。 (√)3.二叉树中每个结点的两棵子树是有序的。 (×)4.二叉树中每个结点有两棵非空子树或有两棵空子树。 (×)5.二叉树中每个结点的关键字值大于其左非空子树(若存在的话)所有结点的关键字值,且小于其右非空子树(若存在的话)所有结点的关键字值。(应当是二叉排序树的特点) (×)6.二叉树中所有结点个数是21-1,其中k是树的深度。(应21)(×)7.二叉树中所有结点,如果不存在非空左子树,则不存在非空右子树。 (×)8.对于一棵非空二叉树,它的根结点作为第一层,则它的第i层上最多能有2i—1个结点。(应21) (√)9.用二叉链表法()存储包含n个结点的二叉树,结点的2n个指针区域中有1个为空指针。 (正确。用二叉链表存储包含n个结点的二叉树,结点共有2n个链域。由于二叉树中,除根结点外,每一个结点有且仅有一个双亲,所以只有1个结点的链域存放指向非空子女结点的指针,还有1个空指针。)即有后继链接的指针仅1个。 (√)10.具有12个结点的完全二叉树有5个度为2的结点。 最快方法:用叶子数=[2]=6,再求n20-1=5

(r ) 11、哈夫曼树中没有度为1的结点,所以必为满二叉树。 (r )12、在哈夫曼树中,权值最小的结点离根结点最近。 (r )13、线索二叉树是一种逻辑结构。 (√)14、深度为K的完全二叉树至少有21个结点。 (√ )15、具有n个结点的满二叉树,其叶结点的个数为(1)/2。 (√ )16、前序和中序遍历用线索树方式存储的二叉树,不必使用栈。 (╳ )17、哈夫曼树是带权路径长度最短的树,路径上权值较大的点离根较远。 二、填空 1.由3个结点所构成的二叉树有 5 种形态。 2. 一棵深度为6的满二叉树有 n12=0+ n2= n0-1=31 个分支结点和 26-1 =32 个叶子。 注:满二叉树没有度为1的结点,所以分支结点数就是二度结点数。3.一棵具有257个结点的完全二叉树,它的深度为 9 。 (注:用2(n) +1= 8 +1=9 4.设一棵完全二叉树有700个结点,则共有 350 个叶子结点。 答:最快方法:用叶子数=[2]=350 5. 设一棵完全二叉树具有1000个结点,则此完全二叉树有 500 个叶子结点,有 499 个度为2的结点,有 1 个结点只有非空左子树,有0 个结点只有非空右子树。 答:最快方法:用叶子数=[2]=500 ,n20-1=499。另外,最后一结点为2i属于左叶子,右叶子是空的,所以有1个非空左子树。完全二叉树的特点决定不可能有左空右不空的情况,所以非空右子树数=0.

相关主题
文本预览
相关文档 最新文档