树与二叉树:数据结构核心解析

引言

在前面的文章中,我们已经系统学习了线性数据结构------链表、栈、队列。线性结构的特点是元素之间存在一对一 的先后关系。然而,现实世界中的很多数据关系是一对多的:文件系统中的目录与子目录、公司的组织架构、网页的 DOM 结构......

树(Tree) 正是用来描述这种层级关系的非线性数据结构。它是整个数据结构体系中最重要的篇章之一,也是后续学习堆、B 树、红黑树、图等高级结构的基础。

第一部分:树的基本概念

一、什么是树

树是由 n(n ≥ 0)个节点组成的有限集合:

  • 当 n = 0 时,称为空树

  • 当 n > 0 时,有一个特殊的节点称为根节点(Root) ,其余节点可分为 m 个互不相交的有限集合,每个集合本身又是一棵树,称为根的子树(Subtree)

二、核心术语

度、深度、高度的对比

三、树的表示方法

1. 双亲表示法(用数组存父节点下标)

cpp 复制代码
#define MAX_NODES 100

typedef struct {
    char data;        // 节点数据
    int parent;       // 父节点下标(根节点的 parent = -1)
} PTNode;

typedef struct {
    PTNode nodes[MAX_NODES];
    int n;            // 节点数量
} PTree;

2. 孩子表示法(每个节点带子节点链表)

cpp 复制代码
typedef struct ChildNode {
    int child_index;           // 子节点在数组中的下标
    struct ChildNode* next;    // 下一个兄弟
} ChildNode;

typedef struct {
    char data;
    ChildNode* first_child;    // 第一个子节点的链表头
} CTBox;

typedef struct {
    CTBox nodes[MAX_NODES];
    int n;
} CTree;

3. 孩子兄弟表示法(最常用,二叉树的基础)

cpp 复制代码
typedef struct CSNode {
    char data;
    struct CSNode* first_child;   // 第一个孩子
    struct CSNode* next_sibling;  // 下一个兄弟
} CSNode;

第二部分:二叉树

一、二叉树的定义

二叉树是每个节点最多只有两个子节点 的树,子节点分为左子节点右子节点

cpp 复制代码
typedef struct BinTreeNode {
    int data;                    // 数据域
    struct BinTreeNode* left;    // 左子节点
    struct BinTreeNode* right;   // 右子节点
} BinTreeNode;

二、两种特殊的二叉树

1. 满二叉树(Full Binary Tree)

每一层的节点数都达到最大值。第 k 层有 2^(k-1) 个节点,总节点数为 2^h - 1(h 为层数)。

2. 完全二叉树(Complete Binary Tree)

叶子节点只出现在最底层和次底层,且最底层的叶子节点都连续集中在左边

完全二叉树的重要特性:可以用数组高效存储。

三、二叉树的性质

性质 内容
性质1 第 i 层最多有 2^(i-1) 个节点(i ≥ 1)
性质2 深度为 k 的二叉树最多有 2^k - 1 个节点
性质3 对于任意二叉树,若叶子数为 n₀,度为 2 的节点数为 n₂,则 n₀ = n₂ + 1
性质4 n 个节点的完全二叉树深度为 ⌊log₂n⌋ + 1

性质3证明

四、二叉树的遍历

遍历是二叉树最基础也是最重要的操作。共有四种遍历方式:

4.1 递归实现
cpp 复制代码
#include <stdio.h>
#include <stdlib.h>

typedef struct TreeNode {
    int data;
    struct TreeNode* left;
    struct TreeNode* right;
} TreeNode;

// 前序遍历:根 → 左 → 右
void preorder(TreeNode* root) {
    if (root == NULL) return;
    printf("%d ", root->data);
    preorder(root->left);
    preorder(root->right);
}

// 中序遍历:左 → 根 → 右
void inorder(TreeNode* root) {
    if (root == NULL) return;
    inorder(root->left);
    printf("%d ", root->data);
    inorder(root->right);
}

// 后序遍历:左 → 右 → 根
void postorder(TreeNode* root) {
    if (root == NULL) return;
    postorder(root->left);
    postorder(root->right);
    printf("%d ", root->data);
}
4.2 层序遍历(BFS,需要队列)
cpp 复制代码
#include <stdbool.h>

#define MAX_QUEUE 100

typedef struct {
    TreeNode* data[MAX_QUEUE];
    int front, rear;
} Queue;

void initQueue(Queue* q) { q->front = q->rear = 0; }
bool isEmpty(Queue* q) { return q->front == q->rear; }
void enqueue(Queue* q, TreeNode* node) { 
    q->data[q->rear++] = node; 
}
TreeNode* dequeue(Queue* q) { 
    return q->data[q->front++]; 
}

// 层序遍历
void levelorder(TreeNode* root) {
    if (root == NULL) return;
    
    Queue q;
    initQueue(&q);
    enqueue(&q, root);
    
    while (!isEmpty(&q)) {
        TreeNode* node = dequeue(&q);
        printf("%d ", node->data);
        
        if (node->left != NULL)  enqueue(&q, node->left);
        if (node->right != NULL) enqueue(&q, node->right);
    }
}

层序遍历过程图解

4.3 非递归实现(需要栈)
cpp 复制代码
// 非递归中序遍历
void inorder_iterative(TreeNode* root) {
    TreeNode* stack[100];
    int top = -1;
    TreeNode* cur = root;
    
    while (cur != NULL || top != -1) {
        // 一直走到最左边
        while (cur != NULL) {
            stack[++top] = cur;
            cur = cur->left;
        }
        // 弹出栈顶,访问它,然后转向右子树
        cur = stack[top--];
        printf("%d ", cur->data);
        cur = cur->right;
    }
}
相关推荐
知南x几秒前
【DPDK核心知识了解】(3) 数据结构与缓冲区管理(建立零拷贝的认知)
数据结构
bIo7lyA8v22 分钟前
算法与数据结构协同优化的设计思想的技术8
数据结构·算法
Lsk_Smion24 分钟前
力扣实训 _ [98].验证二叉搜索树 _ 将二叉树展开成链表
数据结构·算法·leetcode
小小工匠34 分钟前
Redis - 从数据结构到高可用的九个关键问题
数据结构·redis
CHHH_HHH40 分钟前
【C++】红黑树:比AVL树更实用的平衡二叉搜索树
开发语言·数据结构·c++·算法·stl
_日拱一卒1 小时前
LeetCode:17电话号码的字母组合
java·数据结构·算法·leetcode·职场和发展
zzz_23681 小时前
【Redis】Redis 数据结构与 Spring Boot 集成
数据结构·spring boot·redis
Lsk_Smion11 小时前
力扣实训 _ [102].层序遍历--前序--后续_递归与非递归的实现
数据结构·算法·leetcode
Lsk_Smion12 小时前
力扣实训 _ [25].K个一组链表
数据结构·链表
小欣加油12 小时前
leetcode3751 范围内总波动值I
java·数据结构·c++·算法·leetcode