C/C++数据结构之深入了解树与二叉树:概念、存储结构和遍历

news2024/11/23 13:44:48

树是一种常见的数据结构,它在计算机科学和数学中都有广泛的应用。树结构的最简单形式是二叉树,本文将深入探讨树和二叉树的概念、存储结构以及二叉树的遍历,并提供一些实际的代码示例来帮助理解这些概念。

树与二叉树的概念

树 (Tree)

树是一种层次性数据结构,由节点(或称为顶点)和边组成。树的一个特点是它没有环路,即不存在从一个节点出发经过若干边后再回到原节点的路径。树通常包括一个根节点,它没有父节点,以及若干子树,每个子树也是一个树。树中的节点分为内部节点和叶节点,内部节点具有子节点,而叶节点没有子节点。

二叉树 (Binary Tree)

二叉树是一种特殊的树结构,每个节点最多有两个子节点,通常称为左子节点和右子节点。二叉树有多种变体,包括二叉搜索树(Binary Search Tree)和平衡二叉树(Balanced Binary Tree),它们在数据存储和检索方面具有重要的应用。

树与二叉树的存储结构

树和二叉树可以以多种方式进行存储,最常见的方法是使用节点和引用链接的方式。以下是一个简单的二叉树节点的定义:

struct BinaryTreeNode {
    int data;
    BinaryTreeNode* left;
    BinaryTreeNode* right;

    BinaryTreeNode(int value) : data(value), left(nullptr), right(nullptr) {}
};

每个节点包含一个数据元素和指向左子节点和右子节点的指针。通过这种方式,可以递归地构建整个二叉树结构。

二叉树的遍历

树的遍历是指按照一定规则访问树中的所有节点,以便查找、处理或打印它们的值。常用的二叉树遍历方法包括前序遍历、中序遍历和后序遍历。

1. 前序遍历 (Preorder Traversal)

前序遍历从根节点开始,按照根-左-右的顺序遍历节点。以下是递归和非递归的C++代码示例:

递归方式:
void preorderTraversal(BinaryTreeNode* node) {
    if (node != nullptr) {
        cout << node->data << " ";  // 先访问根节点
        preorderTraversal(node->left);  // 再访问左子树
        preorderTraversal(node->right);  // 最后访问右子树
    }
}
非递归方式(使用栈):
#include <stack>

void iterativePreorderTraversal(BinaryTreeNode* root) {
    if (root == nullptr) return;

    stack<BinaryTreeNode*> nodeStack;
    nodeStack.push(root);

    while (!nodeStack.empty()) {
        BinaryTreeNode* current = nodeStack.top();
        nodeStack.pop();
        cout << current->data << " ";  // 访问当前节点

        if (current->right != nullptr) {
            nodeStack.push(current->right);  // 先将右子节点入栈
        }
        if (current->left != nullptr) {
            nodeStack.push(current->left);  // 再将左子节点入栈
        }
    }
}

2. 中序遍历 (Inorder Traversal)

中序遍历按照左-根-右的顺序遍历节点。以下是递归和非递归的C++代码示例:

递归方式:
void inorderTraversal(BinaryTreeNode* node) {
    if (node != nullptr) {
        inorderTraversal(node->left);  // 先访问左子树
        cout << node->data << " ";  // 再访问根节点
        inorderTraversal(node->right);  // 最后访问右子树
    }
}
非递归方式(使用栈):
void iterativeInorderTraversal(BinaryTreeNode* root) {
    stack<BinaryTreeNode*> nodeStack;
    BinaryTreeNode* current = root;

    while (current != nullptr || !nodeStack.empty()) {
        while (current != nullptr) {
            nodeStack.push(current);
            current = current->left;
        }

        current = nodeStack.top();
        nodeStack.pop();
        cout << current->data << " ";  // 访问当前节点
        current = current->right;
    }
}

3. 后序遍历 (Postorder Traversal)

后序遍历按照左-右-根的顺序遍历节点。以下是递归和非递归的C++代码示例:

递归方式:
void postorderTraversal(BinaryTreeNode* node) {
    if (node != nullptr) {
        postorderTraversal(node->left);  // 先访问左子树
        postorderTraversal(node->right);  // 再访问右子树
        cout << node->data << " ";  // 最后访问根节点
    }
}

非递归方式(使用两个栈):
void iterativePostorderTraversal(BinaryTreeNode* root) {
    if (root == nullptr) return;

    stack<BinaryTreeNode*> s1, s2;
    s1.push(root);

    while (!s1.empty()) {
        BinaryTreeNode* current = s1.top();
        s1.pop();
        s2.push(current);

        if (current->left != nullptr) {
            s1.push(current->left);
        }
        if (current->right != nullptr) {
            s1.push(current->right);
        }
    }

    while (!s2.empty()) {
        BinaryTreeNode* current = s2.top();
        s2.pop();
        cout << current->data << " ";  // 访问当前节点
    }
}

示例与分析

考虑以下二叉树:手画不好看,用空格敲出来的。

        1
       / \
      2   3
     / \
    4   5

对该二叉树进行前序、中序和后序遍历的C++代码示例如下:

int main() {
    BinaryTreeNode* root = new BinaryTreeNode(1);
    root->left = new BinaryTreeNode(2);
    root->right = new BinaryTreeNode(3);
    root->left->left = new BinaryTreeNode(4);
    root->left->right = new BinaryTreeNode(5);

    cout << "前序遍历:";
    preorderTraversal(root);
    cout << endl;

    cout << "中序遍历:";
    inorderTraversal(root);
    cout << endl;

    cout << "后序遍历:";
    postorderTraversal(root);
    cout << endl;

    return 0;
}

这些遍历方法将输出以下结果:

前序遍历: 1 2 4 5 3

中序遍历: 4 2 5 1 3

后序遍历: 4 5 2 3 1

换句话说,三种二叉树遍历方式是根节点在不同位置的遍历方式,它们在处理二叉树时具有不同的应用场景和用途。让我们更详细地讨论一下这三种遍历方式:

  1. 前序遍历 (Preorder Traversal):前序遍历从根节点开始,然后按照根-左-右的顺序遍历节点。在前序遍历中,首先访问根节点,然后递归地访问左子树,最后递归地访右子树。这种遍历方式通常用于创建树的副本或表达式求值。

  2. 中序遍历 (Inorder Traversal):中序遍历按照左-根-右的顺序遍历节点。在中序遍历中,首先递归地访问左子树,然后访问根节点,最后递归地访问右子树。这种遍历方式通常用于获取二叉搜索树的元素按升序排列。

  3. 后序遍历 (Postorder Traversal):后序遍历按照左-右-根的顺序遍历节点。在后序遍历中,首先递归地访问左子树,然后递归地访问右子树,最后访问根节点。这种遍历方式通常用于释放树的内存或执行某些计算。

这些C++代码示例清晰展示了如何创建一个二叉树,以及如何使用递归和非递归方式执行前序、中序和后序遍历。这些遍历方法在不同的应用中具有不同的用途,例如前序遍历可以用于复制整个树结构,中序遍历用于获取有序数据,后序遍历常用于计算表达式树。

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

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

相关文章

[双指针] Leetcode 283.移动零和1089.复习零

[双指针] Leetcode 283.移动零和1089.复习零 移动零 283. 移动零 1.题意分析 (1) 给你一个数组&#xff0c;将数组中的所有0移动到数组的末尾 (2) 保证非0元素在数组中相对位置不变 (3) 在原数组中操作 2.解题思路 由于题目要求我们移动数组内容&#xff08;也就是交换两…

【JAVA学习笔记】49 - String类,StringBuffer类,StringBuilder类(重要)

String类 一、String入门 1) String对象用于保存字符串&#xff0c;也就是一组字符序列 2)字符串常量对象是用双引号括起的字符序列。例如: "你好"、 "12.97"、 "boy"等 3)字符串的字符使用Unicode字符编码&#xff0c;一个字符&#xff08;不…

城市群(Megalopolis)/城际(inter-city)OD相关研究即Open Access数据集调研

文章目录 1 城市群/城际OD定义2 理论模型与分析方法2.1 重力模型 Gravity Model2.2 干预机会模型 Intervening Opportunities Model2.3 辐射模型 Radiation Model 3 Issues related to OD flows3.1 OD Prediction3.2 OD Forecasting3.3 OD Construction3.4 OD Estimation 4 OD …

Kubernetes - Ingress HTTP 负载搭建部署解决方案(新版本v1.21+)

在看这一篇之前&#xff0c;如果不了解 Ingress 在 K8s 当中的职责&#xff0c;建议看之前的一篇针对旧版本 Ingress 的部署搭建&#xff0c;在开头会提到它的一些简介Kubernetes - Ingress HTTP 负载搭建部署解决方案_放羊的牧码的博客-CSDN博客 开始表演 1、kubeasz 一键安装…

星闪技术 NearLink 一种专门用于短距离数据传输的新型无线通信技术

本心、输入输出、结果 文章目录 星闪技术 NearLink 一种专门用于短距离数据传输的新型无线通信技术前言星闪技术 NearLink 的诞生背景星闪技术 NearLink 简介星闪技术 NearLink 技术是一种蓝牙技术吗星闪技术 NearLink 优势星闪技术 NearLink 应用前景弘扬爱国精神星闪技术 Nea…

AlmaLinux正开发成为不包含RHEL代码但兼容RHEL的发行版本

导读近日消息&#xff0c;AlmaLinux 正在创建一个不包含 Red Hat Enterprise Linux&#xff08;RHEL&#xff09;代码&#xff0c;但兼容 RHEL 的发行版本。 AlmaLinux OS 基金会主席 benny Vasquez 出席 All Things Open 开源社区大会&#xff0c;表示 AlmaLinux 发行版的目标…

jenkins自动化操作步骤(gitblit)

1、登陆地址&#xff1a; http://xxxxxxxxx.org:xxxx/ admin/xxxx 2、创建任务 选择构建一个maven项目 3、配置 最多只保留一天一个任务 选择git仓库和账号密码 选择代码对应分支 build项&#xff1a; 1&#xff09;使用父项目的pom文件&#xff1a;k56-boot/pom.xml 2&…

什么是鱼叉式网络钓鱼?

鱼叉式网络钓鱼 1. 鱼叉式网络钓鱼的概念2. 鱼叉式网络钓鱼的原理3. 鱼叉式网络钓鱼与网络钓鱼的区别4. 如何防范鱼叉式网络钓鱼 1. 鱼叉式网络钓鱼的概念 鱼叉式网络钓鱼&#xff08;Spear Phishing &#xff09;&#xff0c;又称鱼叉式网络攻击&#xff0c;是一种针对特定目…

SpringCore完整学习教程4,入门级别

本章从第4章开始 4. Logging Spring Boot使用Commons Logging进行所有内部日志记录&#xff0c;但保留底层日志实现开放。为Java Util Logging、Log4J2和Logback提供了默认配置。在每种情况下&#xff0c;记录器都预先配置为使用控制台输出和可选的文件输出。 默认情况下&…

云服务器搭建Spark集群

文章目录 1. Local 模式1.1 安装local模式1.2 命令行工具1.3 提交本地应用 2. Standlone模式2.1 集群配置2.2 修改配置文件2.3 启动集群与停止集群2.4 提交应用到集群环境2.5 提交应用的参数详细说明2.6 配置历史服务2.7 配置高可用&#xff08;HA&#xff09; 3. Yarn模式&…

“爱知道”,你知道吗?

拥抱时代浪潮&#xff0c;加速科技变革。数字经济时代&#xff0c;杭州重点贯彻市委市政府数字经济创新提质“一号发展工程”&#xff0c;加快发展数字经济&#xff0c;推动全市数字经济往高攀升、向新进军、以融提效。基于政府对数字经济新活力的赋能、优化数字社会环节、构建…

『力扣刷题本』:删除排序链表中的重复元素

一、题目 给定一个已排序的链表的头 head &#xff0c; 删除所有重复的元素&#xff0c;使每个元素只出现一次 。返回 已排序的链表 。 示例 1&#xff1a; 输入&#xff1a;head [1,1,2] 输出&#xff1a;[1,2]示例 2&#xff1a; 输入&#xff1a;head [1,1,2,3,3] 输出&am…

守牢底线——建行驻江门市分行纪检组举办2023年清廉合规大讲堂

为推动廉洁教育打通“最后一公里”&#xff0c;近日&#xff0c;建行驻江门市分行纪检组举办江门市分行2023年清廉合规大讲堂。 本次大讲堂邀请了检察院资深检察官专题讲授《金融从业人员易涉犯罪问题剖析及预防》&#xff0c;检察官结合一线办案经历&#xff0c;从防范化解金…

Spring FactoryBean 源码讲解

Spring FactoryBean 源码讲解 什么是Spring FactoryBean Spring FactoryBean是一个特殊的Bean&#xff0c;它实现了FactoryBean接口并重写了其getObject()方法&#xff0c;用于生产其他Bean的实例。在Spring容器启动时&#xff0c;会自动调用FactoryBean的getObject()方法来获…

设计模式:访问者模式(C#、JAVA、JavaScript、C++、Python、Go、PHP)

上一篇《状态模式》 下一篇《原型模式》 简介&#xff1a; 访问者模式&#xff0c;它是一种将数据操作与数据结构分离的设计模式&#xff0c;它属于行为型模式。访问者模式的基本思想是&#xff0c;针对…

Redis中的数据类型以及适用场景

1.Redis中的数据类型 Redis中的数据类型包括&#xff1a;String(字符串&#xff09;、Hash(字典)、List(列表)、Set(集合)、Sorted Set【Zset】(有序集合&#xff09;。 Redis 所有的数据结构都是一个key对应一个value&#xff0c;不同类型的数据结构之间的差异就在于value的…

电子器件 电阻参数与选型

一、参数 电阻的主要参数有&#xff1a;精度、温度系数和功率三个 1.1 精度 一般有0.1%、1%&#xff0c;5%&#xff0c;10%&#xff0c;15%、25%等&#xff0c;一般精度越高价格也越高。有些场合需要使用高精度的电阻。 其中精度所代表的字母如下&#xff1a; L0.01%P0.02…

建行广东江门分行:科技赋能,数据助力纠“四风”

为进一步深化落实中央八项规定精神&#xff0c;持续加大“四风”问题查处力度&#xff0c;建行驻江门市分行纪检组根据《广东省分行贯彻落实中央八项规定精神持之以恒纠治“四风”实施方案》&#xff08;建粤党发〔2023〕1号&#xff09;安排&#xff0c;对驻在市分行开展“四风…

基于ssm实验室管理系统

功能如图所示 摘要 实验室管理系统&#xff08;Laboratory Management System&#xff0c;简称LMS&#xff09;是一种基于SSM&#xff08;Spring、SpringMVC、MyBatis&#xff09;框架的信息化工具&#xff0c;旨在帮助实验室管理者高效、精确地管理实验室资源、人员和实验数据…

详解Java的八种基本数据类型

目录 Java八种数据类型 注意&#xff1a; 四个大类 整型&#xff08;byte、short、int、long&#xff09; 注意 浮点型&#xff08;float、double&#xff09; 神奇的代码 注意&#xff1a; 字符型&#xff08;char&#xff09; 布尔型&#xff08;boolean&#xff0…