第五章题解

第五章 树和二叉树

本章我们重点学习了树和二叉树的定义、二叉树的性质、存储结构、二叉树的先中后序遍历。简略的学习了树和森林的存储结构、遍历,大致讲解了哈夫曼树的基本感念。在学习过程中,建树和二叉树的遍历对我来讲比较难。建树可以用顺序存储也可以用二叉链表存储,二叉树的遍历可以用递归实现也可以用非递归实现。下面主要讲讲我在PTA做题过程的一些感想和思路。

List Leaves

这道题的思路很简单: 构造二叉树的结点结构体、建立二叉树、利用队列输出叶子结点

第一步 构造二叉树的结点结构体

由于这道题的数据N只在10以内,所以我用了顺序存储结构。

1 typedef struct{ //定义二叉树的结点结构 
2     int Left;
3     int Right;
4 }BiTreenode;

第二步 建立二叉树

这里要注意的问题有:

1.输入的是char类型数据,因此为了标号方便要将其ASCII码减去0的ASCII码

2. 要定义一个数组root(bool类型或者int类型),以利于标记这个结点的子树是否是根结点

int Buildtree(BiTreenode T[]){ 
    
    bool root[10]={false};
    int n,i;
    char x,y;
    cin>>n;
    
    for(i=0;i){
        
        cin>>x>>y;
        if(x!='-'){//判断左儿子是否为空,若不为空则左二子不为根结点,赋值为true 
            T[i].Left=x-'0';
            root[T[i].Left]=true;
        } 
        else T[i].Left=-1;
         
        if(y!='-'){ //判断右儿子是否为空,若不为空则右儿子不为根结点,赋值为true 
            T[i].Right=y-'0';
            root[T[i].Right]=true;    
        }
        else T[i].Right=-1;        
    }    
    
    for (int i=0; i//寻找根结点 ,当 root[]为-1时该式子成立,返回i的下标,即根节点的下标 
        if(!root[i])   
        return i;
    }
}

第三步 利用队列输出叶子结点

利用队列的好处是先进先出,而且在push一个数据之后就将它pop出来直接对它的左右孩子进行比较,非常方便

void OrderTree(BiTreenode T[],int x){//利用队列输出叶子结点 
    
    queue<int> q;
    int k;
    q.push(x);//将根节点入队 
    while(!q.empty()){//当队列不为空的时候,继续执行循环 
        
        int tmp=q.front();//访问队列的首结点,也就是将最早入队的结点的下标赋值给tmp 
        
        if(T[tmp].Left!=-1) //若该节点的左儿子不为空,则将它压入队列中 
        q.push(T[tmp].Left);
        if(T[tmp].Right!=-1)//若该节点的右儿子不为空,则将它压入队列中 
        q.push(T[tmp].Right); 
        
        q.pop();   //弹出队列中的第一个元素 
        if(T[tmp].Left==-1 && T[tmp].Right==-1){ //若该结点的左儿子和右儿子均为空,则该结点为叶子结点,输出该叶子结点 
            cout<<tmp;
            if(!q.empty()) //若队列不为空,则输出的不是最后一个叶子结点,在输出空格 
            cout<<" ";
        }    
            
    } 
}    

 

 树的同构

这道题的思路主要是:构造结点结构体、构造二叉树、将两棵二叉树进行比较。

第一步 构造结点结构体

这道题我也是用了顺序存储结构,与上一道题不一样的地方是,这道题需要输入字符域element。

typedef struct Treenode{
    char element;
    int left;
    int right;
}Treenode;

第二步 构造二叉树

与上一道题目大同小异,主要是这里需要构造两棵树。

第三步 将两棵二叉树进行比较

这个步骤有些复杂,主要是在比较过程中要把逻辑思路搞清楚。

1. 两棵树都是空树的情况

2. 其中一棵树是空树的情况下

3. 根节点都存在但数值不同

4. 左子树不存在,则递归判断右子树

5. 左子树同时不空则无需交换左右子树,否则交换左右子树

bool Comparetree(int R1,int R2){
    
    
    if( (R1==Null) && (R2==Null) )
        return true;//两棵树都为空,则同构 
        
    if( ((R1==Null) && (R2!=Null)) || ((R1!=Null) && (R2==Null)) )
        return false;//两棵树中有一棵为空,另外一棵树不为空,则不同构
         
    if( T1[R1].element!=T2[R2].element ) 
        return false;//若两棵树都不为空,但树根的值不相同,则不同构 
        
    if( (T1[R1].left==Null) && (T2[R2].left==Null) ) 
        return Comparetree( T1[R1].right,T2[R2].right );//两棵树都不为空,根的值相同。若左子树都为空,则是否同构取决于右边,递归调用右子树的判别 
        
    if( ((T1[R1].left!=Null) && (T2[R2].left!=Null)) && ((T1[T1[R1].left].element)==(T2[T2[R2].left].element)) ) 
        return(Comparetree(T1[R1].left,T2[R2].left) && Comparetree(T1[R1].right,T2[R2].right));//左子树是否同时不空,若同时不空则无需交换左右子树 
        
    else return (Comparetree(T1[R1].left,T2[R2].right) && Comparetree(T1[R1].right,T2[R2].left));
};

目标总结:

完成了三分之二的题解,比上次进步一点。

下次目标:

完成所有题解。

 

你可能感兴趣的:(第五章题解)