LeetCode 剑指 Offer II 树(下) 专题总结

  • 📚 博客主页:⭐️这是一只小逸白的博客鸭~⭐️
  • 👉 欢迎 关注❤️点赞👍收藏⭐️评论📝
  • 😜 小逸白正在备战实习,经常更新面试题LeetCode题解,欢迎志同道合的朋友互相交流~
  • 💙 若有问题请指正,记得关注哦,感谢~

往期文章 :

先补充一个函数知识点,最后两题会用到:

set、map:

  • lower_bound():二分查找一个有序集合,返回第一个>= num的数,如果没找到,返回末尾的迭代器位置
  • upper_bound():二分查找一个有序集合,返回第一个> num的数,如果没找到,返回末尾的迭代器位置

vector:
在从小到大的排序数组中

  • lower_bound( begin,end,num):从数组的begin位置到end-1位置二分查找第一个 >= num 的数字
  • upper_bound( begin,end,num):从数组的begin位置到end-1位置二分查找第一个 > num 的数字
    在从大到小的排序数组中
  • lower_bound( begin,end,num):从数组的begin位置到end-1位置二分查找第一个 <= num 的数字
  • upper_bound( begin,end,num):从数组的begin位置到end-1位置二分查找第一个 < num 的数字

053. 二叉搜索树中的中序后继

题目:

给定一棵二叉搜索树和其中的一个节点 p ,找到该节点在树中的中序后继。如果节点没有中序后继,请返回 null
节点 p 的后继是值比 p.val 大的节点中键值最小的节点,即按中序遍历的顺序节点 p 的下一个节点。

示例:

在这里插入图片描述
输入:root = [5,3,6,2,4,null,null,1], p = 6
输出:null
解释:因为给出的节点没有中序后继,所以答案就返回 null 了。

提示:

  • 树中节点的数目在范围 [1, 104] 内。
  • -105 <= Node.val <= 105
  • 树中各节点的值均保证唯一。

思路:

用一个res记录大于 p 的节点,每次深搜都会逐渐靠近

  • 当前节点大于目标节点就往左边查找,保存当前节点(当前节点大于目标节点)
  • 否则往右边查找
class Solution {
public:
    TreeNode* res = NULL;
    TreeNode* inorderSuccessor(TreeNode* root, TreeNode* p) {
        dfs(root, p);
        return res;
    }
    void dfs(TreeNode* root, TreeNode* p) {
        if(root == NULL) return ;
        if(root->val > p->val) {
            res = root;
            dfs(root->left, p);
        }else 
            dfs(root->right, p);
    }
};

054. 所有大于等于节点的值之和

题目:

给定一个二叉搜索树,请将它的每个节点的值替换成树中大于或者等于该节点值的所有节点值之和。
提醒一下,二叉搜索树满足下列约束条件:

  • 节点的左子树仅包含键 小于 节点键的节点。
  • 节点的右子树仅包含键 大于 节点键的节点。
  • 左右子树也必须是二叉搜索树。

示例:

在这里插入图片描述
输入:root = [4,1,6,0,2,5,7,null,null,null,3,null,null,null,8]
输出:[30,36,21,36,35,26,15,null,null,null,33,null,null,null,8]

提示:

  • 树中的节点数介于 0104 之间。
  • 每个节点的值介于 -104104 之间。
  • 树中的所有值 互不相同 。
  • 给定的树为二叉搜索树。

思路:

遍历顺序:右 - 中 - 左 , ans记录当前遍历过的节点和,记得更新root->val

class Solution {
public:
    TreeNode* convertBST(TreeNode* root) {
        int ans = 0;
        dfs(root, ans);
        return root;
    }
    //右 - 中 - 左 遍历, ans记录和
    void dfs(TreeNode* root, int& ans) {
        if(root == nullptr) return ;
        dfs(root->right, ans);

        root->val += ans;
        ans = root->val;
        
        dfs(root->left, ans);

    }
};

055. 二叉搜索树迭代器

题目:

实现一个二叉搜索树迭代器类BSTIterator ,表示一个按中序遍历二叉搜索树(BST)的迭代器:

  • BSTIterator(TreeNode root) 初始化 BSTIterator 类的一个对象。BST 的根节点 root 会作为构造函数的一部分给出。指针应初始化为一个不存在于 BST 中的数字,且该数字小于 BST 中的任何元素。
  • boolean hasNext() 如果向指针右侧遍历存在数字,则返回 true ;否则返回 false
  • int next()将指针向右移动,然后返回指针处的数字。
    注意,指针初始化为一个不存在于 BST 中的数字,所以对 next() 的首次调用将返回 BST 中的最小元素。

可以假设 next() 调用总是有效的,也就是说,当调用 next() 时,BST 的中序遍历中至少存在一个下一个数字。

示例:

在这里插入图片描述
输入
inputs = [“BSTIterator”, “next”, “next”, “hasNext”, “next”, “hasNext”, “next”, “hasNext”, “next”, “hasNext”]
inputs = [[[7, 3, 15, null, null, 9, 20]], [], [], [], [], [], [], [], [], []]
输出
[null, 3, 7, true, 9, true, 15, true, 20, false]
解释
BSTIterator bSTIterator = new BSTIterator([7, 3, 15, null, null, 9, 20]);
bSTIterator.next(); // 返回 3
bSTIterator.next(); // 返回 7
bSTIterator.hasNext(); // 返回 True
bSTIterator.next(); // 返回 9
bSTIterator.hasNext(); // 返回 True
bSTIterator.next(); // 返回 15
bSTIterator.hasNext(); // 返回 True
bSTIterator.next(); // 返回 20
bSTIterator.hasNext(); // 返回 False

提示:

  • 树中节点的数目在范围 [1, 105]
  • 0 <= Node.val <= 106
  • 最多调用 105hasNextnext 操作

思路:

将二叉搜索树中序遍历并存储到数组中,通过数组实现next()hasNext() 就简单多了

class BSTIterator {
private:
    vector<int> arr;
    int index = 0;
    //中序遍历并存储到数组中
    void dfs(TreeNode* root) {
        if(!root) return ;
        dfs(root->left);
        arr.push_back(root->val);
        dfs(root->right);
    }

public:
    
    BSTIterator(TreeNode* root) {
        dfs(root);
    }
    
    int next() {
        return arr[index++];
    }
    
    bool hasNext() {
        if(index < arr.size())
            return true;
        else
            return false;
    }
};

056. 二叉搜索树中两个节点之和

题目:

给定一个二叉搜索树的 根节点 root 和一个整数 k , 请判断该二叉搜索树中是否存在两个节点它们的值之和等于 k 。假设二叉搜索树中节点的值均唯一。

示例:

输入: root = [8,6,10,5,7,9,11], k = 12
输出: true
解释: 节点 5 和节点 7 之和等于 12

提示:

  • 二叉树的节点个数的范围是 [1, 104].
  • -104 <= Node.val <= 104
  • root 为二叉搜索树
  • -105 <= k <= 105

思路:

将遍历过的值存储在 map 中,通过map[k - root->val]map中寻找答案

class Solution {
public:
    bool ok = false;
    unordered_map<int, int> map;
    bool findTarget(TreeNode* root, int k) {
        dfs(root, k);
        return ok;
    }
    void dfs(TreeNode* root, int k) {
        if(root == nullptr || ok) return ;
        if(map[k - root->val]) {
            ok = true;
            return ;
        }
        map[root->val]++;
        dfs(root->left, k);
        dfs(root->right, k);
    }
};

057. 值和下标之差都在给定的范围内

题目:

给你一个整数数组 nums 和两个整数 kt 。请你判断是否存在 两个不同下标 ij,使得 abs(nums[i] - nums[j]) <= t ,同时又满足 abs(i - j) <= k 。
如果存在则返回 true,不存在返回 false。

示例:

输入:nums = [1,2,3,1], k = 3, t = 0
输出:true

提示:

  • 0 <= nums.length <= 2 * 104
  • -231 <= nums[i] <= 231 - 1
  • 0 <= k <= 104
  • 0 <= t <= 231 - 1

思路:

用滑动窗窗口来约束一定在 k 内
abs(a - b) <= t 情况下, b肯定在[a-t, b+t] (t>0) 区间内
所以求出上限,下限,然后通过 lower_bound() 求出大于等于下限的数,再判断是否小于等于上限即可

class Solution {
public:
    bool containsNearbyAlmostDuplicate(vector<int>& nums, int k, int t) {
        set<long> set;
        for (int i = 0; i < nums.size(); ++i) {
            //下限
            long lowerLimit = (long)nums[i] - t;
            //上限
            long upperLimit = (long)nums[i] + t;
            //查找大于等于下限的数
            auto it = set.lower_bound(lowerLimit);
            if (it != set.end() && *it <= upperLimit) {
                return true;
            }
            set.insert(nums[i]);
            //滑动窗口
            if (i >= k) {
                set.erase(nums[i - k]);
            }
        }
        return false;
    }
};

058. 日程表

题目:

请实现一个 MyCalendar 类来存放你的日程安排。如果要添加的时间内没有其他安排,则可以存储这个新的日程安排。
MyCalendar 有一个 book(int start, int end)方法。它意味着在 start 到 end 时间内增加一个日程安排,注意,这里的时间是半开区间,即 [start, end), 实数 x 的范围为, start <= x < end。
当两个日程安排有一些时间上的交叉时(例如两个日程安排都在同一时间内),就会产生重复预订。
每次调用 MyCalendar.book方法时,如果可以将日程安排成功添加到日历中而不会导致重复预订,返回 true。否则,返回 false 并且不要将该日程安排添加到日历中。
请按照以下步骤调用 MyCalendar 类: MyCalendar cal = new MyCalendar(); MyCalendar.book(start, end)

示例:

输入:
[“MyCalendar”,“book”,“book”,“book”]
[[],[10,20],[15,25],[20,30]]
输出: [null,true,false,true]
解释:
MyCalendar myCalendar = new MyCalendar();
MyCalendar.book(10, 20); // returns true
MyCalendar.book(15, 25); // returns false ,第二个日程安排不能添加到日历中,因为时间 15 已经被第一个日程安排预定了
MyCalendar.book(20, 30); // returns true ,第三个日程安排可以添加到日历中,因为第一个日程安排并不包含时间 20

提示:

  • 每个测试用例,调用 MyCalendar.book 函数最多不超过 1000次。
  • 0 <= start < end <= 109

思路:

题目一堆废话,就是要判断区间是否交叉
要判断当前区间是否能够存入 map ,满足一下两个条件就可以

  • lower_bound() 求出大于等于start的数,这个数的start 要大于 插入数 的end
  • lower_bound()数的前一个数,这个数的 end 要小于插入数的 start
class MyCalendar {
public:
    map<int, int> map;
    MyCalendar() {

    }
    
    bool book(int start, int end) {
        // 查找第一个大于等于start的元素
        auto iter = map.lower_bound(start);
        // (iter)第一个大于等于插入元素的时间段。这个时间段的起始时间不能小于插入时间的终止时间
        if(iter != map.end() && iter->first < end) 
            return false;
        // (--iter)最后一个小于等于插入元素的时间段。这个时间段的终止时间不能大于插入的起始时间
        if(iter != map.begin() && (--iter)->second > start)
            return false;
        // 保存时间段
        map[start] = end;
        return true;
    }
};

  • 21
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 16
    评论
评论 16
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一只小逸白

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值