递归实现二叉树的前、中、后序遍历

什么是二叉树?

二叉树是一种特殊的树形数据结构,它的每个节点最多只能有两个子节点,通常被称为左子节点和右子节点。这个定义非常直观,形象地描述了二叉树的形状。然而,这只是二叉树的表面特征。

首先,二叉树的每个节点都有一个数据元素。这个元素可以是任何类型的数据,如数字、字符甚至是其他的数据结构。这些数据元素存储在节点的左子节点和右子节点中,这使得二叉树成为一个非常灵活的数据存储方式。

其次,二叉树的性质决定了它的应用范围非常广泛。比如,我们可以利用二叉树进行高效的搜索和排序。由于每个节点最多只有两个子节点,因此我们可以很容易地遍历整个树,找到我们需要的元素。此外,二叉搜索树和AVL等高级数据结构也都是在二叉树的基础上构建的。

然而,尽管二叉树有许多优点,但也尤其局限性。例如,当节点的子节点数量过多时,二叉树的搜索效率会降低。这就需要我们在实际应用中,根据具体的需求和场景,选择最合适的数据结构。

总的来说,二叉树是一种简单而强大的数据结构,它为我们提供了一种有效的组织和处理数据的方式。通过深入学习和理解二叉树,可以更好地理解数据结构的原理,进一步提高我们的编程技能。

二叉树的定义

在C++中,可以使用结构体的方式定义一个二叉树(来自LeetCode):

C++ 复制代码
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */

在JavaScript中,可以使用函数来定义一个二叉树(来自LeetCode):

JavaScript 复制代码
/**
 * Definition for a binary tree node.
 * function TreeNode(val, left, right) {
 *     this.val = (val===undefined ? 0 : val)
 *     this.left = (left===undefined ? null : left)
 *     this.right = (right===undefined ? null : right)
 * }
 */

二叉树的前、中、后序遍历指的是访问根节点的顺序。比如前序遍历是优先访问根节点,再遍历左子树,最后遍历右子树;中序遍历则是优先遍历左子树,再访问根节点,最后遍历右子树;同理,后序遍历是先遍历左子树,然后遍历右子树,最后访问根节点。

二叉树的前序遍历

前序遍历的顺序是:根节点->左子树->右子树。

递归实现的方式是,如果二叉树非空,则先访问根节点,然后递归地前序遍历左子树和右子树。

C++ 复制代码
class Solution {
public:
    vector<int> preorderTraversal(TreeNode* root) {
        vector<int> result;
        preorder(root, result);

        return result;
    }

    void preorder(TreeNode* root, vector<int> &result) {
        if (!root) return;

        result.push_back(root->val);
        preorder(root->left, result);
        preorder(root->right, result);
    }
};
JavaScript 复制代码
/**
 * @param {TreeNode} root
 * @return {number[]}
 */
var preorderTraversal = function(root) {
    const result = [];

    const preOrder = function(root) {
        if (!root) return;

        result.push(root.val);
        preOrder(root.left);
        preOrder(root.right);
    }

    preOrder(root);

    return result;
};

二叉树的中序遍历

中序遍历的顺序是:左子树->根节点->右子树。

递归实现的方式是,如果二叉树非空,则先递归地中序遍历左子树,然后访问根节点,最后递归地中序遍历右子树。

C++ 复制代码
class Solution {
public:
    vector<int> inorderTraversal(TreeNode* root) {
        vector<int> result;
        inorder(root, result);

        return result;
    }

    void inorder(TreeNode* root, vector<int> &result) {
        if (!root) return;

        inorder(root->left, result);
        result.push_back(root->val);
        inorder(root->right, result);
    }
};
JavaScript 复制代码
/**
 * @param {TreeNode} root
 * @return {number[]}
 */
var inorderTraversal = function(root) {
    const result = [];

    const inOrder = function(root) {
        if (!root) return;

        inOrder(root.left);
        result.push(root.val);
        inOrder(root.right);
    }

    inOrder(root);

    return result;
};

二叉树的后序遍历

后序遍历地的顺序是:左子树->右子树->根节点。

递归实现的方式是,如果二叉树非空,则先递归地后序遍历左子树,然后递归地后序遍历右子树,最后访问根节点。

C++ 复制代码
class Solution {
public:
    vector<int> postorderTraversal(TreeNode* root) {
        vector<int> result;
        postorder(root, result);
        
        return result;
    }

    void postorder(TreeNode* root, vector<int> &result) {
        if (!root) return;

        postorder(root->left, result);
        postorder(root->right, result);
        result.push_back(root->val);
    }
};
JavaScript 复制代码
/**
 * @param {TreeNode} root
 * @return {number[]}
 */
var postorderTraversal = function(root) {
    const result = [];

    const postOrder = function(root) {
        if (!root) return;

        postOrder(root.left);
        postOrder(root.right);

        result.push(root.val);
    }

    postOrder(root);

    return result;
};

一维数组转二叉树

一维数组和二叉树是两种常见的数据结构,各有其独特的特性和用途。然而,有时我们需要在这两种数据结构之间进行转换。

首先,我们来理解这两种数据结构的基本概念。一维数组是一种线性的数据结构,可以通过索引快速访问任意元素;而二叉树则是一种层次结构的数据结构,每个节点最多有两个子节点,通常称为左子节点和右子节点。

假设我们有一个一维数组,其中包含n个元素,可以按照以下步骤将其转换为二叉树:

  1. 创建一个空的二叉树;
  2. 从一维数组的第一个元素开始,将其作为根节点放入二叉树中;
  3. 对于一维数组中的下一个元素,如果其值小于根节点的值,则将其作为左子节点放入左子树中;否则,将其作为右子节点放入右子树中;
  4. 重复步骤3,直到处理完一维数组中的所有元素。
JavaScript 复制代码
function TreeNode(val, left, right) {
     this.val = (val === undefined) ? 0 : val;
     this.left = (left === undefined) ? null : left;
     this.right = (right === undefined) ? null : right;
}

function buildBinaryTree(arr) {
    var root = null;
    if (!arr || arr.length === 0) return root;
    root = new TreeNode(arr[0]);
    for (var i = 1; i < arr.length; i++) {
        insertNode(arr[i], root);
    }
    
    return root;
}

function insertNode(current, root) {
    if (root === null) root = { val: current, left: null, right: null };
    else if (current <= root.val) root.left = insertNode(current, root.left);
    else root.right = insertNode(current, root.right);
    
    return root;
}
相关推荐
秋说4 分钟前
【数据结构 | C++】整型关键字的平方探测法散列
数据结构·c++·算法
weixin_4786897635 分钟前
【回溯法】——组合总数
数据结构·python·算法
戊子仲秋1 小时前
【LeetCode】每日一题 2024_11_14 统计好节点的数目(图/树的 DFS)
算法·leetcode·深度优先
TaoYuan__2 小时前
机器学习的常用算法
人工智能·算法·机器学习
用户40547878374823 小时前
深度学习笔记 - 使用YOLOv5中的c3模块进行天气识别
算法
十七算法实验室3 小时前
Matlab实现麻雀优化算法优化随机森林算法模型 (SSA-RF)(附源码)
算法·决策树·随机森林·机器学习·支持向量机·matlab·启发式算法
黑不拉几的小白兔3 小时前
PTA部分题目C++重练
开发语言·c++·算法
迷迭所归处3 小时前
动态规划 —— dp 问题-买卖股票的最佳时机IV
算法·动态规划
chordful4 小时前
Leetcode热题100-32 最长有效括号
c++·算法·leetcode·动态规划
_OLi_4 小时前
力扣 LeetCode 459. 重复的子字符串(Day4:字符串)
算法·leetcode·职场和发展·kmp