C语言——线索二叉树(前序、中序、后序-附代码)

news2025/1/23 1:05:34

 一、什么是线索二叉树

线索二叉树(Threaded Binary Tree)是一种特殊的二叉树,通过将空指针改为线索(即前驱或后继指针)的方式,将二叉树中的空闲指针利用起来,从而实现对二叉树的高效遍历和查找。

线索化是将二叉树以某种遍历方式进行扫描,并为每个节点添加线索的过程。

  • 线索二叉树的节点定义包含两个标志位,分别表示其左右指针是否为线索。
  • 如果左孩子节点为空,则将其左孩子指针指向该节点的直接前驱节点;
  • 如果右孩子节点为空,则将其右孩子指针指向该节点的直接后继节点。

这样,在线索化之后,可以通过线索快速定位节点的前驱和后继,避免了不必要的遍历,提高了查找效率。

线索二叉树的好处主要在于它可以提高对二叉树的遍历效率。

常规二叉树的遍历需要使用递归或堆栈等数据结构,效率较低。而线索二叉树通过将原本为空的指针改为前驱或后继线索,实现了对二叉树的高效遍历和查找。

同时,线索二叉树可以有效地节约存储空间,避免了在节点中存储额外的空指针,提高了内存利用率。

二、全部代码(内有详细注释)

2.0、文件结构与代码示例

需要注意的是,因为指针是按值传递的,所以在函数中如果需要修改指针变量的值(如pNode),必须使用指向指针的指针或指针的引用等方式,否则修改不会生效。 在代码中,由于函数没有返回值,因此无法判断函数是否执行成功。如果出现错误或异常情况,程序可能会发生意外的行为或崩溃。

代码中构建的二叉树

 

2.1、main.cpp

#include "test.h"
int main() {

    char *str = "123##45##6##7#8##";     // 前序构造二叉树
    BThead binTreeHead;
    initBinTree(&binTreeHead,'#');

    createBinTreeNode(&binTreeHead,str);  // 前序构造二叉树

    ForEachBinTree(&binTreeHead,1);    // 遍历二叉树(1:前序,2:中序,3:后序)
    threadBinTree(&binTreeHead, 1);    // 线索二叉树(1:前序,2:中序,3:后序)


    printf("%p",&binTreeHead);
}

2.2、test.h

#pragma once

#include<stdio.h>
#include<malloc.h>
#include<assert.h>
#include <math.h>
#include <iostream>

//元素类型
#define ElemType char
#define MyType int


// 二叉树结点
typedef struct BinTreeNode{
    // 标志  0代表是孩子  1代表是前驱或后继
    MyType L_tag;
    MyType R_tag;

    // 节点
    struct BinTreeNode *L_child;
    struct BinTreeNode *R_child;

    // 值
    ElemType value;
}BTnode, *BT;

// 二叉树的头结点
typedef struct BinTreeHead{
    ElemType endFlag;
    BinTreeNode *BTNode;
}BThead;



void initBinTree(BinTreeHead *head, ElemType endFlag);  // 头节点初始化

void createBinTreeNode(BinTreeHead *head, char *str);  // 接收 头结点  一个字符串地址

void ForEachBinTree(BinTreeHead *head,int i); // 遍历二叉树(1:前序,2:中序,3:后序)

void threadBinTree(BinTreeHead *head, int i); // 线索二叉树(1:前序,2:中序,3:后序)

2.3、test.cpp


#include"test.h"
// 定义数据类型
struct BinTreeNode *pre = NULL;

// 申请结点
BinTreeNode* mallocNode(ElemType e){
    BinTreeNode *newNode = (BinTreeNode*)malloc(sizeof(BinTreeNode));
    assert(newNode != NULL);
    newNode->L_child = newNode->R_child = NULL;
    newNode->L_tag = newNode->R_tag = 0;  //线索化之前左右标记都初始为指针链
    newNode->value = e;
    return newNode;
}

// 头节点初始化
void initBinTree(BinTreeHead *head, ElemType endFlag){
    head->endFlag = endFlag;
    head->BTNode = NULL;

}

// 创建二叉树(前序)
void createBTreeNode(BinTreeHead *pHead, BinTreeNode **pNode, char **pString) {

//    printf("%d \n",**pString);
//    printf("%d \n",pHead->endFlag);
    // 如果终止:
    if(**pString == pHead->endFlag){
        *pNode = NULL;
        return;
    }

    // 创建新的根结点
    *pNode = mallocNode(**pString);

    // 创建左孩子
    (*pString)++;
    createBTreeNode(pHead, &((*pNode)->L_child), pString);
    // 创建右孩子
    (*pString)++;
    createBTreeNode(pHead, &((*pNode)->R_child), pString);

}

// 创建二叉树的头结点
void createBinTreeNode(BinTreeHead *head, char *str){
    createBTreeNode(head, &(head->BTNode), &str);
}

//---------------------------------------遍历二叉树
// 输出二叉树
void visit(BinTreeNode *pNode){

    printf("value: %c \n",pNode->value);

}

// 前序遍历二叉树
void preOrder(BinTreeNode *pNode){

    if(pNode == NULL){
        return;
    }

    visit(pNode);  // 输出结点value

    preOrder(pNode->L_child);
    preOrder(pNode->R_child);
}

// 中序遍历二叉树
void inOrder(BinTreeNode *pNode){

    if(pNode == NULL){
        return;
    }

    inOrder(pNode->L_child);
    visit(pNode);  // 输出结点value
    inOrder(pNode->R_child);
}
// 后序遍历二叉树
void afterOrder(BinTreeNode *pNode){

    if(pNode == NULL){
        return;
    }

    afterOrder(pNode->L_child);
    afterOrder(pNode->R_child);
    visit(pNode);  // 输出结点value
}

// 用头结点遍历二叉树
void ForEachBinTree(BinTreeHead *head,int i){
    // 如果是1,则前序遍历
    if(i==1){
        preOrder(head->BTNode);
        return;
    }
    // 如果是2,则中序遍历
    if(i==2){
        inOrder(head->BTNode);
        return;
    }
    // 如果是3,则后序遍历
    if(i==3){
        afterOrder(head->BTNode);
        return;
    }
    printf("请输入对应的数字!");
}
//-------------------------------------------------------------- 线索二叉树

// 线索化
void createThread(BinTreeNode **pNode){


    // 如果左孩子为空      (前驱)
    if((*pNode)->L_child == NULL && (*pNode)->L_tag == 0){
        (*pNode)->L_tag = 1;
        (*pNode)->L_child = pre;
    }

    // 如果pre的右孩子为空  (后继)
    if(pre->R_child == NULL && (*pNode)->R_tag == 0){
        pre->R_tag = 1;
        pre->R_child = *pNode;
    }

    // 回退之前 pre 赋值 当前结点
    pre = (*pNode);

}

// 前序
void preThreadBinTree(BinTreeNode **pNode){

    if(*pNode == NULL){
        return;
    }
    createThread(&(*pNode));  // 线索化
    if((*pNode)->L_tag == 0){    // 如果没有被线索化 !!! --------防止转圈 !!!
        preThreadBinTree(&((*pNode)->L_child));
    }
    preThreadBinTree(&((*pNode)->R_child));

}

// 中序
void inThreadBinTree(BinTreeNode **pNode){
    if(*pNode == NULL){
        return;
    }
    inThreadBinTree(&((*pNode)->L_child));
    createThread(&(*pNode));  // 输出结点value
    inThreadBinTree(&((*pNode)->R_child));

}

// 后序
void afterThreadBinTree(BinTreeNode **pNode){
    if(*pNode == NULL){
        return;
    }

    preThreadBinTree(&((*pNode)->L_child));
    preThreadBinTree(&((*pNode)->R_child));
    createThread(&(*pNode));  // 输出结点value
}


// 创建线索二叉树
void threadBinTree(BinTreeHead *head, int i){
    // pre初始化 - 让其指向根结点
    pre = head->BTNode;

    // 如果是1,则前序 创建线索二叉树
    if(i==1){
        preThreadBinTree(&(head->BTNode));
        return;
    }
    // 如果是2,则中序 创建线索二叉树
    if(i==2){
        inThreadBinTree(&(head->BTNode));
        return;
    }
    // 如果是3,则后序 创建线索二叉树
    if(i==3){
        afterThreadBinTree(&(head->BTNode));
        return;
    }
    printf("请输入对应的数字!");
}



三、注意事项

构造线索二叉树时,需要注意以下几点:

  1. 在线索化过程中,需要保证已经线索化的节点不能再次线索化,否则会形成死循环。

  2. 由于线索化操作会改变原有的指针关系,因此需要在构造完成后进行还原,以便保证二叉树结构的完整性。

  3. 如果通过线索来遍历二叉树,需要考虑如何恰当地使用前驱和后继线索,以避免遍历过程中出现错误。

在进行前序、中序、后序构造时,需要注意以下几点:

  1. 通过前序遍历和中序遍历,可以唯一确定一个二叉树。因此,在进行构造时,需要保证输入的前序和中序序列是正确的,且序列中的元素没有重复。

  2. 对于中序遍历和后序遍历,同样也可以唯一确定一个二叉树。在进行构造时,需要满足输入的中序和后序序列是正确的,且序列中的元素没有重复。

  3. 在实际操作中,需要注意递归的边界情况,以避免出现死循环或者溢出等异常情况。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/464323.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

go源码解读-sync.pool

go version 1.19.7 sync.pool 是go 内置的对象池技术&#xff0c; 管理临时对象&#xff0c;这些对象可以单独保存和检索&#xff0c; 减少GC次数 特点&#xff1a;1、 池不可以指定大小 2、 Get 没有的话会新生成一个对象 3、对象的周期取决于GC的周期 从go doc可以看到sync.p…

13、MDK分散加载方式管理多块内存

MDK分散加载: 默认情况下是通过MDK的option选项设置Flash和RAM大小&#xff0c;这种情况下所有的管理工作都是编译来处理的&#xff0c; MDK自动生成的分散加载文件&#xff1a;H7_ProjectTest.sct ; ************************************************************* ; *…

Java_异常

Java_异常 1.什么是异常 ​ 生活中的异常&#xff1a;感冒发烧、电脑蓝屏、手机死机等。 ​ 程序中的异常&#xff1a;磁盘空间不足、网络连接中断、被加载的资源不存在等。 ​ 程序异常解决办法&#xff1a;针对程序中非正常情况&#xff0c;Java语言引入了异常&#xff0…

【C++】类和对象(1)

文章目录 前言浅浅了解一、面向过程和面向对象二、 类和对象的关系三、创建类和对象 逐步深入一、类的访问限定符二、 封装三、类的作用域四、类对象模型五、this指针 前言 浅浅了解 一、面向过程和面向对象 C语言是面向过程的&#xff0c;关注的是过程&#xff0c;分析出求解…

智能汽车开启中央计算革命,全场景智能“车芯”强势崛起

伴随着汽车跨域融合时代的到来&#xff0c;智能汽车芯片正处于快速迭代期&#xff0c;同时牌桌上的玩家也在加速挪换位置。 一方面&#xff0c;包括丰田、大众集团等在内的全球汽车制造商正在进入芯片平台的切换周期&#xff0c;加速推动汽车芯片市场格局的改变。 另一方面&a…

Ubuntu22.04部署eurekaserver集群

Ubuntu22.04部署eurekaserver集群 为了更好的浏览体验&#xff0c;欢迎光顾勤奋的凯尔森同学个人博客http://www.huerpu.cc:7000 每次都启动eureka的项目&#xff0c;太繁琐了&#xff0c;我们把eureka部署到Ubuntu&#xff0c;就可以愉快的玩耍了。 1 配置文件设置 准备了…

设计模式 -- 观察者模式

前言 月是一轮明镜,晶莹剔透,代表着一张白纸(啥也不懂) 央是一片海洋,海乃百川,代表着一块海绵(吸纳万物) 泽是一柄利剑,千锤百炼,代表着千百锤炼(输入输出) 月央泽,学习的一种过程,从白纸->吸收各种知识->不断输入输出变成自己的内容 希望大家一起坚持这个过程,也同…

淘宝天猫数据分析:2023年健康养生三大品类数据分析

随着人们健康意识的不断增强&#xff0c;越来越多的年轻人都开始加入养生大军的队伍中&#xff0c;我国的健康养生产业也迎来了发展机遇。 在天猫平台上&#xff0c;养生茶、养生壶和滋补养生原料是养生市场的几大重点类目&#xff0c;接下来&#xff0c;结合鲸参谋电商数据分析…

Docker 相关概念

1、Docker是什么&#xff1f; 如何确保应用能够在这些环境中运行和通过质量检测&#xff1f;并且在部署过程中不出现令人头疼的版本、配置问题&#xff0c;也无需重新编写代码和进行故障修复&#xff1f; 答案就是使用容器。Docker之所以发展如此迅速&#xff0c;也是因为它对…

电脑硬盘分区合并怎么操作?分享2个方法!

案例&#xff1a;电脑硬盘怎么分区&#xff1f; 【我把我的电脑硬盘分成了多个区域&#xff0c;这样可以方便存储和管理数据。现在我需要调整分区&#xff0c;对分区进行合并&#xff0c;但我不知道该如何操作&#xff0c;有没有小伙伴知道&#xff1f;】 在使用电脑的过程中…

4核8G云服务器4c8g或4h8g指的是什么?

4核8G云服务器什么意思&#xff1f;4c8g或4h8g代表CPU内存配置&#xff0c;4c8g是指4核CPU、8G内存&#xff0c;准确来讲由于是云服务器&#xff0c;4核指的是4核vCPU&#xff0c;4核8G就是指云服务器CPU内存配置。云服务器不只是CPU内存&#xff0c;还有公网带宽和系统盘&…

经典 Learned Index 结构设计及其应用

引言 学习索引是一种新型的索引结构&#xff0c;可以帮助数据库更快地查找数据。学习索引的诞生可以追溯到 2017 年&#xff0c;由 Google Brain 团队的 Kraska 等人在论文[1]中首次提出,探讨了使用神经网络替代传统数据结构&#xff08;如 B-Tree&#xff09;来构建索引的可行…

appuploader 常规使用登录方法

转载&#xff1a;登录appuploader 目录 登录appuploader 常规使用登录方法 双击appuploader.exe 启动appuploader 点击底部的未登录&#xff0c;弹出登录框 在登录框内输入apple开发者账号 如果没有apple开发者账号&#xff0c;只是普通的apple账号&#xff0c;请勾选上未…

题目 2056: 汉诺塔 ==理解递归

题目 2056: 汉诺塔 https://www.dotcpp.com/oj/problem2056.html 做题情况 参考代码&#xff1a; //package Dotcpp;import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);int n sc.nextInt();// prin…

3.30 haas506 2.0开发教程-example - SD卡存储数据读写

SD卡存储数据读写 案例说明数据的写入与读取串口工具读取数据接收数据CSV格式 案例说明 部分设备使用过程中需要保存大量数据到TF卡中&#xff0c;大部分场景拔插TF卡有不太方便。 所以本案例介绍一种使用串口工具取出设备TF卡中的数据保存在电脑中的方法。 保存格式可以自己定…

详细安装使用教程】店侦探 - 跟踪店铺数据,学习运营技巧,引流关键词,电商人必备工具

简介 店侦探插件是一款电商网络浏览插件&#xff0c;能够帮助店主更好地运营自己的网店&#xff0c;这款插件功能十分全面强大&#xff0c;可以全面跟踪店铺的销量情况、引流关键词、直通车、营销活动、宝贝变更跟踪&#xff01;感兴趣的朋友快来体验吧&#xff0c;跟踪店铺数…

常见的用户密码加密及破解方法

用户密码安全是互联网行业需要保障的重要安全之一&#xff0c;由于黑客的入侵和内部的泄露&#xff0c;保证用户密码安全并不是件容易的事情&#xff0c;但如果采用合适的算法加密用户密码&#xff0c;即使信息泄露出去&#xff0c;黑客也无法还原出原始的密码(或者还原的代价非…

vue3 封装ECharts组件

一、前言 前端开发需要经常使用ECharts图表渲染数据信息&#xff0c;在一个项目中我们经常需要使用多个图表&#xff0c;选择封装ECharts组件复用的方式可以减少代码量&#xff0c;增加开发效率。 ECharts图表大家应该用的都比较多&#xff0c;基础的用法就不细说了&#xff…

如何成为企业急需的技术人才:掌握这些技能,提升你的实力和竞争力

在当前竞争激烈的互联网环境中&#xff0c;作为程序员等技术岗&#xff0c;必须不断的学习&#xff0c;才能不断提升自身实力&#xff0c;锻炼自身技能。想要成为一名企业急需的技术人才&#xff0c;需要学习哪些技能呢&#xff1f; 一、IT技术发展背景及历程 IT技术是当今社…

如何借助分布式存储 JuiceFS 加速 AI 模型训练

传统的机器学习模型&#xff0c;数据集比较小&#xff0c;模型的算法也比较简单&#xff0c;使用单机存储&#xff0c;或者本地硬盘就足够了&#xff0c;像 JuiceFS 这样的分布式存储并不是必需品。 随着近几年深度学习的蓬勃发展&#xff0c;越来越多的团队开始遇到了单机存储…