数据结构——链式二叉树(C语言版)

链式二叉树的结构

⽤链表来表⽰⼀棵⼆叉树,即⽤链来指⽰元素的逻辑关系。 通常的⽅法是链表中每个结点由三个域组成,数据域和左右指针域,左右指针分别⽤来给出该结点左孩⼦和右孩⼦所在的链结点的存储地址。
                                
由上图可见,链式二叉树的每个节点存有三个变量,第一个为节点存的数据,第二个与第三个分别为指向左孩子指针与指向右孩子指针。

创建链式二叉树

        由于普通的二叉树并没有插入删除的意义,所以在创建二叉树时只需要手动创建二叉树就行,因为普通二叉树没有条件限制比如:

所以在普通二叉树里并没有增删的意义,创建时只需要手动创建即可。

        创建二叉树的第一步:先写一个创建单个节点的函数,并将节点里左右指针都初始化为NULL,这样也方便后续链接操作。 第二步:手动申请7个节点,并将将节点之间进行链接。最终形成以下的逻辑图片。

        因为根结点的左⼦树和右⼦树分别⼜是由⼦树结点、⼦树结点的左⼦树、⼦树结点的右⼦树组成的,因此⼆叉树定义是递归式的,

链式二叉树的遍历

        前序遍历:

前序遍历(Preorder Traversal 亦称先序遍历):访问根结点的操作发⽣在遍历其左右⼦树之前,及访问顺序为:根,左子树,右子树。就以上图为例子
                        
第一步:先判断此时的根是否为NULL,如果是就输出NULL后返回,第二步:先输出根节点的值,接着在递归根的左子树,第三步:将根的左子树递归完后接着递归右子树。
                                                  左子树展开图
        以上是根节点的左子树的展开图:一开始会先节点是否为空,接着输出根节点数值,接着递根的左子树,当左子树为空的时候就开始递根的右子树,右子树又开始递 根,左子树,右子树,当左右子树都为NULL时候开始归。
当左子树归完后根节点开始递根节点的右子树。
                                                  右子树展开图
最终递归的输出顺序为:

        中序遍历:

        中序遍历(Inorder Traversal):访问根结点的操作发⽣在遍历其左右⼦树之中(间)及访问顺序为:左子树,根,右子树。
                                        
                                                  左子树展开图
        根据顺序,从根开始并没有先输出值而是先开始递归根的左子树,从上图的左子树的递归展开图中,当节点递到3的左子树(NULL)时,先输出NULL接着开始归,重新归到3节点时输出3,接着递归3的右子树(7),并重复步骤,当1的左子树递归完后就输出1节点的值并执行递归1的右子树。
                                                  右子树展开图
        
        当递归1的右子树(4)时,4又作为根开始递归4的左子树与右子树,但遇到NULL节点时就开始进行返回。
        最终中序遍历完成会输出:

        后序遍历:

        后序遍历(Postorder Traversal):访问根结点的操作发⽣在遍历其左右⼦树之后 ,及访问顺序为:左子树,右子树,根。
        
                                                        左子树展开图
                
                                                  右子树展开图
        从后续展开图中可以看出,二叉树的后序遍历,是优先访问根节点的左子树与右子树,将根节点的左右子树访问完最后再访问根节点的值,最终输出的顺序为:
        二叉树的前中后序其实访问的顺序是一样的,只是在于访问根节点数据的时机不同,产生的结果也不同。

        层序遍历:

除了二叉树的先序遍历、中序遍历、后序遍历外,还可以对⼆叉树进⾏层序遍历。设⼆叉树的根结点所在层数为1,层序遍历就是从所在⼆叉树的根结点出发,⾸先访问第⼀层的树根结点,然后从左到右访问第2 上的结点,接着是第三层的结点,以此类推,⾃上⽽下,⾃左⾄右逐层访问树的结点的过程就是层序遍历。
        从上图可以看出,通过队列的特性,先进先出,先将根节点入队,入队后出队,并在出队的时候同时将根的左右子树入队,当节点为NULL则不入队,当队列重新为空时,则结束循环,并打印出二叉树的层序遍历形式。

链式二叉树的简单算法题

        求二叉树所有节点个数:

                                                左子树展开图

                                        右子树展开图

从上图的左右子树展开图图可以看出,求节点根数需要分别递归计算根的左子树与右子树,当递归到NULL时返回0,当左右子树递归完后需要再加上1(自身),再进行返回。最终计算出该子树节点数量为7。

        求二叉树叶子节点的个数:

                                        递归展开图

        因为叶子节点的特点是左右子树都为NULL,利用这一特性,当递到根的左右子树都为NULL,就返回1表示这个节点是叶子节点。如果左节点或又节点其中一个是NULL并不会返回1,而是继续递归。当递归到叶子节点时直接返回一。最终得得出根的左子树节点数与根的右子树节点数。

        求二叉树第k层的节点个数:

                                               

                                                        递归展开图

        由上图的展开图可以看出,假设k==3,分别递归根的左子树与右子树的第k层,每一次递归的时候都让k-1,当k==1的时候就说明已经到了第k层直接返回1,当递归到第k-1层的时,只需要判断k-1层的左节点与右节点,如果为NULL就返回0,最后将k层返回的数字进行相加并return就得到第k层节点数。

        求二叉树的深度:

                                        左子树深度展开图

                                右子树深度展开图        

        想要计算二叉树的深度,就要计算根的左子树与右子树的深度,将左子树的深度与右子树的深度进行比较,并返回较大值,如上图,通过递归展开,会得知左子树的深度为3,右子树的深度为2,最终返回左子树的深度+1,所以上图二叉树的深度为4。

        二叉树查找值为x的节点:

                                递归展开图

        由上图可见,当需要找x节点时,一开始会先判断根节点是否符合条件,如果符合条件直接返回,如果不符合就先递归根的左子树,当左子树递归完后发现并没有找到值会返回NULL,此时指针为NULL,会递归根的右子树,当发现有节点符合值的时候会直接返回并给指针赋值,防止继续递归。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值