【面试HOT200】二叉树的构建二叉搜索树篇

news2024/11/29 10:52:46

系列综述:
💞目的:本系列是个人整理为了秋招面试的,整理期间苛求每个知识点,平衡理解简易度与深入程度。
🥰来源:材料主要源于【CodeTopHot200】进行的,每个知识点的修正和深入主要参考各平台大佬的文章,其中也可能含有少量的个人实验自证,所有代码均优先参考最佳性能。
🤭结语:如果有帮到你的地方,就点个赞关注一下呗,谢谢🎈🎄🌷!!!
🌈【C++】秋招&实习面经汇总篇


文章目录

    • 二叉树的构建
      • 基础知识
        • 654. 构建二叉树*
      • 相关题目
        • 105. 从前序与中序遍历序列构造二叉树
        • 106. 从中序与后序遍历序列构造二叉树
    • 二叉搜索树(考察较少,留后处理)
      • 基础知识
      • 相关题目
        • 98. 验证二叉搜索树
        • 530. 二叉搜索树的最小绝对差
        • 236. 二叉树的最近公共祖先
        • 235. 二叉搜索树的最近公共祖先
        • 450. 删除二叉搜索树中的节点
        • 669. 修剪二叉搜索树
        • 108. 将有序数组转换为二叉搜索树
        • 669. 修剪二叉搜索树
    • 参考博客


😊点此到文末惊喜↩︎

二叉树的构建

基础知识

654. 构建二叉树*
  1. 654. 最大二叉树
    • 通过始末位置指示容器范围,避免每次调用的vector创建开销
TreeNode* constructMaximumBinaryTree(vector<int>& nums) {
    auto self = [&](auto &&self, int left, int right)->TreeNode*{
        // 递归出口
        if (left > right) return nullptr;
        // 如何划分:查找区间中的最大根节点
        int max_pos = left;
        for (int i = left+1; i <= right; ++i) {
            if (nums[i] > nums[max_pos]) 
                max_pos = i;
        }
        // 建立根节点,左递归,右递归
        TreeNode *root = new TreeNode(nums[max_pos]);
        root->left = self(self, left, max_pos-1);
        root->right = self(self, max_pos+1, right);
        // 返回根节点
        return root;
    };
    TreeNode *root = self(self, 0, nums.size()-1);
    return root;
}

相关题目

105. 从前序与中序遍历序列构造二叉树
  1. 105. 从前序与中序遍历序列构造二叉树

    在这里插入图片描述
TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder){
    auto self = [&](auto &&self, 
    int pre_left, int pre_right, int in_left, int in_right)->TreeNode*{
    	// 递归出口:只需要前序的左右指针符合即可
        if (pre_left > pre_right) return nullptr;
		
        // 1. 将前序序列的第一个结点作为根节点
        TreeNode *root = new TreeNode(preorder[pre_left]);
        // 2. 找到前序序列的第一个结点在后序序列中的位置作为划分 
        int pivot = in_left; 
        for (; pivot <= in_right; ++pivot){ // key:注意初始化
            if (inorder[pivot] == preorder[pre_left]) //  key:查找中序的pivot
                break;
        }
        // 3. 计算前序数组中的划分位置
        int pre_pivot = pre_left + pivot - in_left;
        // 4. 建立二叉树
        root->left = self(self, pre_left+1, pre_pivot, in_left, pivot-1);
        root->right = self(self, pre_pivot+1, pre_right, pivot+1, in_right);
		
        return root;
    };
    return self(self, 0, preorder.size()-1, 0, inorder.size()-1);
}
106. 从中序与后序遍历序列构造二叉树
  1. 106. 从中序与后序遍历序列构造二叉树

    • 通过始末位置指示容器范围,避免每次调用的vector创建开销
TreeNode* buildTree(vector<int>& inorder, vector<int>& postorder) {
    auto self = [&](auto &&self, 
    int in_left, int in_right, int post_left, int post_right)->TreeNode*{
        // 递归出口
        if (post_left > post_right) return nullptr;
        // 建立根结点
        TreeNode *root = new TreeNode(postorder[post_right]);
        // 在中序数组中查找划分位置
        int pivot = in_left;
        for (; pivot <= in_right; ++pivot) {
            if (inorder[pivot] == postorder[post_right])
                break;
        }
        // 建立左右子树
        int post_pivot = post_right - (in_right - pivot);// key:注意括号
        root->left = self(self, in_left, pivot-1, post_left, post_pivot-1);
        root->right = self(self, pivot+1, in_right, post_pivot, post_right-1);

        return root;
    };
    return self(self, 0, inorder.size()-1, 0, postorder.size()-1);
}

二叉搜索树(考察较少,留后处理)

基础知识

相关题目

98. 验证二叉搜索树
  1. 98. 验证二叉搜索树
    • 中序遍历下,输出的二叉搜索树节点的数值是有序序列
    // **********中序遍历,形成一个递增数组**************
    vector<int> vec;
    void inorder(TreeNode *root){
        if(root == nullptr) return ;
        inorder(root->left);
        vec.push_back(root->val);
        inorder(root->right);
    }
    // 判断是否中序遍历的数组是递增的
    bool isValidBST(TreeNode* root){
        inorder(root);
        for(int i = 0; i < vec.size()-1; ++i){
            if(vec[i] >= vec[i+1])// 二叉搜索树的中序排列是严格递增的
                return false;
        }
        return true;
    }
    
    // *********************纯递归**********************
    bool isValid(TreeNode* current,long left,long right){
        // 单层逻辑
        if(current==nullptr) 
            return true;
        else if(current->val<=left||current->val>=right) 
            return false;
        // 递归
        return isValid(current->left,left,current->val)
        		&&isValid(current->right,current->val,right);
    }
    bool isValidBST(TreeNode* root) {
        return isValid(root,LONG_MIN,LONG_MAX);
    }
    

530. 二叉搜索树的最小绝对差
  1. 530. 二叉搜索树的最小绝对差
    • 思路:中序遍历下,输出的二叉搜索树节点的数值是有序序列。顺序判断相邻值的绝对值,保存最小的即可
    • 双指针在树内应用,双指针本质是对于一个序列的遍历。
    int getMinimumDifference(TreeNode* root) {
        // 初始化条件
        stack<TreeNode*> st;
        if(root != nullptr) st.push(root);
        int res = INT_MAX;
        TreeNode *prior = new TreeNode(-1000000);
    
        while(!st.empty()){
            TreeNode* cur = st.top();
            if(cur != nullptr){
                st.pop();
    			// 中序遍历
                if(cur->right) st.push(cur->right);
                st.push(cur);
                st.push(nullptr);
                if(cur->left) st.push(cur->left);
            }else{
                st.pop();
                cur = st.top();
                st.pop();
                // 节点处理
                res = min(res, cur->val - prior->val);
                prior = cur;// 迭代条件
                
            }
        }
        return res;
    }
    

236. 二叉树的最近公共祖先
  1. 236. 二叉树的最近公共祖先
    • 后序遍历是一个天然的自低向上的回溯过程
    • 状态的向上传递:通过判断左右子树是否出现了p和q,如果出现p或q则通过回溯值上传到父节点
        TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
            if(root == NULL)
                return NULL;
            // 每次对返回的结点进行
            if(root == p || root == q) 
                return root;
                
            TreeNode* left =  lowestCommonAncestor(root->left, p, q);
            TreeNode* right = lowestCommonAncestor(root->right, p, q);
           	// 结点的处理是:尽量返回结点
            if(left == NULL)
                return right;
            if(right == NULL)
                return left;      
            if(left && right) // p和q在两侧
                return root;
            
            return NULL; // 必须有返回值
        }
    

235. 二叉搜索树的最近公共祖先
  1. 235. 二叉搜索树的最近公共祖先
    • 思路:自上而下搜索,遇到的第一个节点值在p和q之间的值即为最近公共祖先
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        while(root) {
            if (root->val > p->val && root->val > q->val) {
                root = root->left;
            } else if (root->val < p->val && root->val < q->val) {
                root = root->right;
            } else return root;
        }
        return NULL;
    }
    

450. 删除二叉搜索树中的节点
  1. 450. 删除二叉搜索树中的节点
    • 思路:框架
    TreeNode* deleteNode(TreeNode* root, int key) {
        // 健壮性检查
        if(root == nullptr) return nullptr;
        // 基本初始化
        TreeNode *cur = root;
        TreeNode *prior = root;
        while (cur != nullptr){
            // 符合条件值的处理
            if(cur->val == key){
                if(cur->left == nullptr || cur->right == nullptr){
                    // 两个都空
                    if(cur->left == nullptr && cur->right == nullptr) 
                        return nullptr;
                    // 被删除节点只有一个孩子或均为空
                    if(key < prior->val){// cur是左子树
                        prior->left = cur->right;
                        return root;  
                    }else{
                        prior->right n = cur->right;
                        return root; 
                    }
                }else{
                    // 被删除节点有两个孩子
                    TreeNode *curLeft = cur->left;
                    cur = cur->right;
                    while(cur->left != nullptr){
                        cur = cur->left;
                    }
                    cur->left = curLeft;
    
                    if(key < prior->val){// cur是左子树
                        prior->left = prior->left->right;
                        return root;  
                    }else{
                        prior->right = prior->right->right;
                        return root; 
                    }
                    
                    
                }
                
            }
    
            prior = cur;// 前迭代
            // 左右节点处理
            if(key < cur->val){
                if(cur->left){
                    cur = cur->left;
                }else{// 找不到
                    return root;
                }
            }else{
                if(cur->right){
                    cur = cur->right;
                }else{// 找不到
                    return root;
                }
            }
            
        }
    
        return root;
    
    }
    

669. 修剪二叉搜索树
  1. 669. 修剪二叉搜索树
    // 1. 确定递归函数的返回类型及参数,返回类型是递归算法的输出值类型,参数是递归算法的输入
    TreeNode* trimBST(TreeNode* root, int low, int high) {
        // 2. 递归终止条件
        if (root == nullptr ) return nullptr;
    
        // 3.节点处理:return保留的状态
        if (root->val < low) {// 保留更大的右半部分
            TreeNode* right = trimBST(root->right, low, high);
            return right;
        }
        if (root->val > high) {// 保留更小的左半部分
            TreeNode* left = trimBST(root->left, low, high); 
            return left;
        }
    
        // 4.迭代条件
        root->left = trimBST(root->left, low, high); // root->left接入符合条件的左孩子
        root->right = trimBST(root->right, low, high); // root->right接入符合条件的右孩子
        return root;
    }
    

108. 将有序数组转换为二叉搜索树
  1. 108. 将有序数组转换为二叉搜索树
    TreeNode* traversal(vector<int>& nums, int left, int right) {
        // 递归出口
        if (left > right) return nullptr;
        // 运算
        int mid = left + ((right - left) / 2);// 防止求和溢出
        TreeNode* root = new TreeNode(nums[mid]);
        // 递归迭代
        root->left = traversal(nums, left, mid - 1);
        root->right = traversal(nums, mid + 1, right);
        return root;
    }
    // 主调函数
    TreeNode* sortedArrayToBST(vector<int>& nums) {
        TreeNode* root = traversal(nums, 0, nums.size() - 1);
        return root;
    }
    

669. 修剪二叉搜索树
  1. 669. 修剪二叉搜索树
    // 1. 确定递归函数的返回类型及参数,返回类型是递归算法的输出值类型,参数是递归算法的输入
    TreeNode* trimBST(TreeNode* root, int low, int high) {
        // 2. 递归终止条件
        if (root == nullptr ) return nullptr;
    
        // 3.节点处理:return保留的状态
        if (root->val < low) {// 保留更大的右半部分
            TreeNode* right = trimBST(root->right, low, high);
            return right;
        }
        if (root->val > high) {// 保留更小的左半部分
            TreeNode* left = trimBST(root->left, low, high); 
            return left;
        }
    
        // 4.迭代条件
        root->left = trimBST(root->left, low, high); // root->left接入符合条件的左孩子
        root->right = trimBST(root->right, low, high); // root->right接入符合条件的右孩子
        return root;
    }
    

少年,我观你骨骼清奇,颖悟绝伦,必成人中龙凤。
不如点赞·收藏·关注一波

🚩点此跳转到首行↩︎

参考博客

  1. 「代码随想录」47. 全排列 II:【彻底理解排列中的去重问题】详解
  2. codetop

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

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

相关文章

免费SSL证书靠谱吗?

首先&#xff0c;我们需要明确一点&#xff0c;那就是免费SSL证书并非完全没有价值。它们同样能够提供基本的数据加密功能&#xff0c;确保用户的信息在传输过程中不会被第三方截获。因此&#xff0c;如果你的网站不需要处理敏感信息&#xff0c;例如个人身份信息、银行卡号等&…

地方招商策略:招商招哪些,如何选择理想的企业?

招商引资是推动地方经济发展的不二选择&#xff0c;通过吸引优质企业入驻&#xff0c;不仅可以带来直接的投资和税收&#xff0c;还可以为地方创造更多的就业机会&#xff0c;引入高端人才、先进的技术及管理经验&#xff0c;同时&#xff0c;招商引资还能够促进地方的产业升级…

深度优先搜索(DFS)LeetCode 2477. 到达首都的最少油耗

2477. 到达首都的最少油耗 给你一棵 n 个节点的树&#xff08;一个无向、连通、无环图&#xff09;&#xff0c;每个节点表示一个城市&#xff0c;编号从 0 到 n - 1 &#xff0c;且恰好有 n - 1 条路。0 是首都。给你一个二维整数数组 roads &#xff0c;其中 roads[i] [ai,…

回溯总结(一)基础概念及模板

1.回溯是什么&#xff1f; 回溯&#xff0c;也叫回溯搜索法&#xff0c;搜索的一种方式。回溯搜索实际上也是一种暴力搜索&#xff08;本质是穷举&#xff09;&#xff08;对于有些问题是唯一可以解决的办法了&#xff0c;for循环是不适用的&#xff09;和别的搜索不同之处在于…

基于混沌算法的图像加密解密系统

1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 研究背景与意义&#xff1a; 随着信息技术的迅猛发展&#xff0c;图像的传输和存储已经成为现代社会中不可或缺的一部分。然而&#xff0c;随着互联网的普及和信息的快速传播&am…

Node-red的节点离线安装

Node-red节点离线安装 前言 目前越来越高的数据安全的要求&#xff0c;因此我们很多的生产类服务器是无法进行在线化部署的&#xff0c;为了解决这一问题&#xff0c;我们需要在无法连接外部网络环境的情况下&#xff0c;实现Node-red的节点的安装&#xff0c;以满足项目的需…

el-dialog 垂直居中

写文章总是在想引言&#xff0c;怎么开头才会显的更加优雅&#xff0c;更加让读者朋友给我点赞。看到有人点赞&#xff0c;我就觉的进行技术经验分享是一件非常愉快的事情&#xff0c;可是打小作文写的不好不会组织语句&#xff0c;就喜欢直来直去。老师说让写春天的作文&#…

vue2 element-ui select下拉框 选择传递多个参数

<el-select v-model"select" slot"prepend" placeholder"请选择" change"searchPostFn($event,123)"> <el-option :label"item.ziDianShuJu" :value"{value:item.id, label:item.ziDianShuJu}" v-for&qu…

文件夹批量改名:轻松管理文件夹,随机重命名不求人

在日常生活和工作中&#xff0c;文件夹批量改名是一个常见的需求。当有大量的文件夹时&#xff0c;它们可能会变得混乱和难以管理。有时候要对大量的文件夹进行重命名&#xff0c;以便更好地组织和管理这些文件。手动重命名每个文件夹可能会非常耗时且容易出错。现在可以来看下…

印度股市荣登全球第四,释放了什么讯号?

KlipC报道&#xff1a;随着散户迅速增加&#xff0c;外国资金重新流入&#xff0c;自2020年3月大流行低点以来&#xff0c;截至当地时间12月4日&#xff0c;其印度股市总市值已经上涨到3.93万亿美元&#xff0c;仅次于美国、中国和日本。 值得一提的是&#xff0c;Sensex指数飙…

java springboot简单了解数据源实现 与 springboot内置数据源

之前 我们讲到的项目 数据库管理 用了三种技术 数据源管理方式 我们选择了: DruidDataSource 持久化技术: MyBatis-Plus / MyBatis 数据库: MySql 那么 我们在刚接触数据库连接时 是没用配置Druid的 那它有没有用数据源呢&#xff1f; 我们接触过的配置Druid的方式有两种 用…

CSS特效026:扇骨打开关闭的动画

CSS常用示例100专栏目录 本专栏记录的是经常使用的CSS示例与技巧&#xff0c;主要包含CSS布局&#xff0c;CSS特效&#xff0c;CSS花边信息三部分内容。其中CSS布局主要是列出一些常用的CSS布局信息点&#xff0c;CSS特效主要是一些动画示例&#xff0c;CSS花边是描述了一些CSS…

xcode ——Instrumets(网络连接调试)使用

环境&#xff1a; instruments 使用只能在真机调试时使用&#xff0c;且真机系统必须ios15 点击debug 按钮——Network——Profile in Instruments 然后就可以看到如下面板 展开运行的项目&#xff0c;点击session下的域名&#xff0c;下方回出现该域名下的网络请求。点击Deve…

持续集成交付CICD:Jenkins使用GitLab共享库实现前后端项目Sonarqube

目录 一、实验 1.Jenkins使用GitLab共享库实现后端项目Sonarqube 2.优化GitLab共享库 3.Jenkins使用GitLab共享库实现前端项目Sonarqube 二、问题 1.sonar-scanner 未找到命令 2.npm 未找到命令 一、实验 1.Jenkins使用GitLab共享库实现后端项目Sonarqube &#xff08…

现代版“田忌赛马”:IPMT如何实现资源利用最大化

摘要&#xff1a;在流程调整和产品重整过程中&#xff0c;IPMT发挥了至关重要的作用。IPMT是什么&#xff1f;它的职责到底是什么&#xff1f;每个公司必须要有IPMT吗&#xff1f; 1992年&#xff0c;IBM在激烈的市场竞争下&#xff0c;遭遇到了严重的财政困难&#xff0c;利润…

用js自定义一个(v-model)vModel双向绑定函数

vue中的v-model是双向绑定的, 我们自己用JavaScript实现一个双向绑定vModel函数。 // element 元素或者#id,.class,div 得是input标签 // data 对象 // 将要绑定property 对象中的key<input class"vmodel"/>function vModel(element, data, property) {if (…

外汇天眼:假冒、非法经营成常态?超60家外汇平台被拉黑

近期&#xff0c;全球范围内多个国家的金融监管机构纷纷发出警告&#xff0c;揭露一系列假冒、非法经营的外汇交易平台。比利时金融服务和市场管理局&#xff08;FSMA&#xff09;发现53家外汇平台涉嫌非法运营&#xff0c;而意大利CONSOB和英国金融行为监管局&#xff08;FCA&…

【Redis】Redis 的学习教程(十三)Redis 各场景

由于Redis 支持比较丰富的数据结构&#xff0c;因此他能实现的功能并不仅限于缓存&#xff0c;而是可以运用到各种业务场景中&#xff0c;开发出既简洁、又高效的系统 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-bo…

Ribbon 饥饿加载

Ribbon默认是采用懒加载&#xff0c;即第一次访问时才会去创建LoadBalanceClient&#xff0c;请求时间会很长而饥饿加载则会在项目启动时创建&#xff0c;降低第一次访问的耗时&#xff0c;通过下面配置开启饥饿加载: 一、懒加载 Ribbon 默认为懒加载即在首次启动Application…

防火墙 iptables的使用

目录 什么是防火墙 原理 代理 防火墙的工具 4表5列 五链&#xff1a;控制流量的时机 四个表&#xff1a;如何控制流量 ​编辑 iptables 软件 格式 选项 跳转 查iptables 的规则 添加规则 A I 删除规则 清空规则 替换规则 R 修改默认规则&#xff08;默…