当前位置: 首页 > article >正文

力扣第95题 不同的二叉搜索树 II

不同的二叉搜索树 II

  • 一级目录
    • 二级目录
      • 三级目录
  • 力扣第95题 - 不同的二叉搜索树 II
      • 题目描述
      • 思路分析
        • 1. 二叉搜索树的性质
        • 2. 递归构造树
        • 3. 动态规划优化(可选)
    • 递归详细
      • 递归函数定义
        • 参数含义
        • 返回值
      • 递归的逻辑
      • 递归过程的可视化
        • 第一次递归(范围 `[1, 3]`)
        • 第二次递归(范围 `[1, 1]` 和 `[3, 3]`)
        • 第三次递归组合(根节点 `2`)
        • 递归退出条件
      • 完整流程
      • 递归优势
      • 算法实现
        • C语言实现
      • 关键点解析
      • 复杂度分析
      • 输出示例
        • 输入:
        • 输出:

一级目录

二级目录

三级目录

力扣第95题 - 不同的二叉搜索树 II

题目描述

给你一个整数 n,要求生成所有由 1n 为节点所组成的 不同的二叉搜索树,并返回这些树的根节点。


思路分析

1. 二叉搜索树的性质
  • 左子树的所有节点值小于根节点值。
  • 右子树的所有节点值大于根节点值。
  • 1n 的每个数字作为根节点时,左右子树的节点范围可以分别确定。
2. 递归构造树
  • generateTrees(start, end) 表示生成从 startend 的所有 BST:
    • 如果 start > end,返回 [NULL](空树)。
    • 遍历 istartend,将 i 作为根节点:
      • 左子树:递归调用 generateTrees(start, i - 1)
      • 右子树:递归调用 generateTrees(i + 1, end)
    • 将所有可能的左子树和右子树组合,形成以 i 为根的所有树。
3. 动态规划优化(可选)
  • 利用备忘录保存 [start, end] 的计算结果,避免重复计算。

递归详细

递归函数定义

struct TreeNode** generateTreesHelper(int start, int end, int* returnSize)
参数含义
  1. start: 当前生成树的节点值范围的起始值。
  2. end: 当前生成树的节点值范围的结束值。
  3. returnSize: 用于返回生成的树的数量。
返回值

返回所有可能的 BST(以数组形式表示,数组中的每个元素是 BST 的根节点指针)。


递归的逻辑

  1. 递归结束条件(Base Case)
    if (start > end) {
        *returnSize = 1;
        struct TreeNode** result = (struct TreeNode**)malloc(sizeof(struct TreeNode*));
        result[0] = NULL; // 空树
        return result;
    }
    
    • start > end 时,不存在可用节点,这表示一种有效的情况:返回空树。
    • 这种情况允许更高层的递归在左右子树组合时将 NULL 作为子树。

  1. 枚举每个节点作为根节点
    for (int i = start; i <= end; i++) {
    
    • 遍历 [start, end] 范围内的每个数字,将其作为当前树的根节点。

  1. 递归生成左右子树
    int leftSize = 0;
    struct TreeNode** leftTrees = generateTreesHelper(start, i - 1, &leftSize);
    
    int rightSize = 0;
    struct TreeNode** rightTrees = generateTreesHelper(i + 1, end, &rightSize);
    
    • 左子树: 节点值范围为 [start, i-1]
    • 右子树: 节点值范围为 [i+1, end]
    • 对于每个可能的根节点 i,递归生成其所有可能的左子树和右子树。

  1. 组合左右子树
    for (int l = 0; l < leftSize; l++) {
        for (int r = 0; r < rightSize; r++) {
            struct TreeNode* root = createNode(i);
            root->left = leftTrees[l];
            root->right = rightTrees[r];
    
            result = (struct TreeNode**)realloc(result, sizeof(struct TreeNode*) * (totalSize + 1));
            result[totalSize++] = root;
        }
    }
    
    • 对于每个可能的左子树和右子树,组合成一棵以 i 为根的完整树:
      • leftTrees[l] 是某种可能的左子树。
      • rightTrees[r] 是某种可能的右子树。
    • 创建一个以 i 为值的新根节点 root
    • leftTrees[l] 连接到 root->left,将 rightTrees[r] 连接到 root->right
    • 将组合完成的树加入结果数组 result 中。

  1. 释放递归中动态分配的内存
    free(leftTrees);
    free(rightTrees);
    
    • 左右子树数组在当前递归结束后不再需要,及时释放内存,避免内存泄漏。

  1. 返回结果
    *returnSize = totalSize;
    return result;
    
    • 将生成的树的数量保存在 returnSize 中。
    • 返回所有可能的 BST。

递归过程的可视化

n = 3 为例,调用 generateTrees(3, &returnSize)

第一次递归(范围 [1, 3]
  • 枚举根节点:1, 2, 3
  • 例如,当根节点为 2
    • 左子树范围: [1, 1]
    • 右子树范围: [3, 3]

第二次递归(范围 [1, 1][3, 3]
  • 对于左子树 [1, 1]
    • 唯一根节点为 1,左右子树均为空。
  • 对于右子树 [3, 3]
    • 唯一根节点为 3,左右子树均为空。

第三次递归组合(根节点 2
  • 左子树为 [1]
  • 右子树为 [3]
  • 组合成以 2 为根的树:
        2
       / \
      1   3
    

递归退出条件
  • 当范围无效(如 [4, 3]),返回空树 [NULL]

完整流程

递归通过分解问题,将大范围 [1, n] 的树拆分成多个小范围 [start, end] 的子树,最终组合成所有可能的二叉搜索树。


递归优势

  • 分治思想: 将问题拆解为左右子树独立生成的子问题,最后组合。
  • 灵活性: 能动态处理不同的 n,适用于范围大小不定的树生成问题。
  • 简洁性: 通过递归和迭代组合,代码结构清晰易读。

算法实现

C语言实现
#include <stdio.h>
#include <stdlib.h>

// 定义树的结构
struct TreeNode {
    int val;
    struct TreeNode* left;
    struct TreeNode* right;
};

// 创建一个新节点
struct TreeNode* createNode(int val) {
    struct TreeNode* node = (struct TreeNode*)malloc(sizeof(struct TreeNode));
    node->val = val;
    node->left = NULL;
    node->right = NULL;
    return node;
}

// 生成从 start 到 end 的所有 BST
struct TreeNode** generateTreesHelper(int start, int end, int* returnSize) {
    if (start > end) {
        *returnSize = 1;
        struct TreeNode** result = (struct TreeNode**)malloc(sizeof(struct TreeNode*));
        result[0] = NULL;
        return result;
    }

    int totalSize = 0;
    struct TreeNode** result = NULL;

    for (int i = start; i <= end; i++) {
        // 左子树
        int leftSize = 0;
        struct TreeNode** leftTrees = generateTreesHelper(start, i - 1, &leftSize);

        // 右子树
        int rightSize = 0;
        struct TreeNode** rightTrees = generateTreesHelper(i + 1, end, &rightSize);

        // 组合左右子树和当前根节点
        for (int l = 0; l < leftSize; l++) {
            for (int r = 0; r < rightSize; r++) {
                struct TreeNode* root = createNode(i);
                root->left = leftTrees[l];
                root->right = rightTrees[r];

                result = (struct TreeNode**)realloc(result, sizeof(struct TreeNode*) * (totalSize + 1));
                result[totalSize++] = root;
            }
        }

        free(leftTrees);
        free(rightTrees);
    }

    *returnSize = totalSize;
    return result;
}

// 主函数:生成从 1 到 n 的所有 BST
struct TreeNode** generateTrees(int n, int* returnSize) {
    if (n == 0) {
        *returnSize = 0;
        return NULL;
    }
    return generateTreesHelper(1, n, returnSize);
}

// 测试用例打印树(先序遍历)
void printTree(struct TreeNode* root) {
    if (!root) {
        printf("null ");
        return;
    }
    printf("%d ", root->val);
    printTree(root->left);
    printTree(root->right);
}

int main() {
    int returnSize;
    struct TreeNode** trees = generateTrees(3, &returnSize);

    printf("Generated %d unique BSTs:\n", returnSize);
    for (int i = 0; i < returnSize; i++) {
        printTree(trees[i]);
        printf("\n");
    }

    return 0;
}

关键点解析

  1. 递归终止条件:当 start > end 时返回空树 [NULL]
  2. 组合左右子树:
    • 遍历 startend,每个数都可能是根节点。
    • 对每个根节点,递归生成左右子树并组合。
  3. 内存管理:
    • 动态分配数组用于保存不同的树。
    • 组合树时需考虑 realloc 的正确使用。

复杂度分析

  1. 时间复杂度
    • 递归中,每次会分成多个子问题,时间复杂度近似为生成不同 BST 的卡特兰数 C n C_n Cn
      C n = 1 n + 1 ( 2 n n ) C_n = \frac{1}{n+1} \binom{2n}{n} Cn=n+11(n2n)
      因此时间复杂度为 O ( C n ) O(C_n) O(Cn)
  2. 空间复杂度
    • 递归栈的深度为 O ( n ) O(n) O(n)
    • 动态分配的内存和树的组合数量相关,为 O ( C n ) O(C_n) O(Cn)

输出示例

输入:
n = 3
输出:
Generated 5 unique BSTs:
1 null 2 null 3
1 null 3 2 null
2 1 null 3 null
3 1 null null 2
3 2 null null 1

这表示 5 种不同的二叉搜索树的先序遍历结果。


http://www.kler.cn/a/429836.html

相关文章:

  • OpenStack 网络服务的插件架构
  • Cline(原Claude Dev)开源的IDE AI插件,如何搭配OpenRouter实现cursor功能,Cline怎么使用
  • 【WebRTC】Android SDK使用教学
  • 如何使用靜態IP代理?【詳細教程】
  • 云原生周刊:在Docker上部署大语言模型
  • scala 身份证号码
  • hping3-网络工具
  • Brain.js(六):构建FNN神经网络实战教程 - 用户喜好预测
  • 鸿蒙实现后台任务管理
  • 「Mac畅玩鸿蒙与硬件44」UI互动应用篇21 - 随机励志语录生成器
  • Springboot3整合Redis
  • Fyne ( go跨平台GUI )中文文档-Fyne总览(二)
  • gitlab配置调试minio
  • Java将数组转换成字符串
  • 构建万能 MOCK-API
  • 如何在拉丁美洲推广游戏
  • docker逃逸总结
  • vue+elementUI+transition实现鼠标滑过div展开内容,鼠标划出收起内容,加防抖功能
  • docker搭建elasticsearch服务
  • python爬虫--小白篇【爬虫实践】
  • R 语言科研绘图第 4 期 --- 折线图-置信区间
  • 一种基于通义千问prompt辅助+Qwen2.5-coder-32b+Bolt.new+v0+Cursor的无代码对话网站构建方法