【线段树】个人练习-Leetcode-3161. Block Placement Queries

题目链接:https://leetcode.cn/problems/block-placement-queries/description/

题目大意:在x轴上,有两种操作:

  • 在某个点上插入一个障碍物
  • 查询,在[0, x]区间内,能否放下长度为sz的物品

插入和查询操作交替进行,返回每次查询的结果。

思路:一开始时看见题目写的"anywhere"以为是【区间内每处都能放下物品】,寻思着那不是整个区间都没有障碍物才行吗,好像判断有点简单了。后来才发现只要有一处能放下就行了。

那其实就是查询[0, x]区间内,被障碍物隔开的各个区间中,能存储的最大长度。这种涉及区间查询的,应该用线段树来解决。然而长久不写已经忘了线段树怎么写了...看了很多题解,还是灵神的比较清晰。

更新i处的值为val

cpp 复制代码
	void update(int o, int l, int r, int i, int val) {
        if (l == r) {
            tree[o] = val;
            return;
        }
        int mid = (l + r) / 2;
        if (i <= mid)
            update(o * 2, l, mid, i, val);
        else
            update(o * 2 + 1, mid+1, r, i, val);
        tree[o] = max(tree[o * 2], tree[o*2 + 1]);
    }

返回[0, x]中最大的区间长度

cpp 复制代码
	int query(int o, int l, int r, int x) {
        if (r <= x)
            return tree[o];
        int mid = (l + r) / 2;
        if (x <= mid) 
            return query(o * 2, l, mid, x);
        return max(tree[o * 2], query(o * 2 + 1, mid + 1, r, x)); 
    }

对于[0, x]来说,要么最大长度在[0, pre]prex左边的最大的障碍物位置)之间,要么在[pre, x]之间(因为每个query中x其实也相当于一个暂时的障碍)。

完整代码

cpp 复制代码
class Solution {
public:
    vector<int> tree;

    void update(int o, int l, int r, int i, int val) {
        if (l == r) {
            tree[o] = val;
            return;
        }
        int mid = (l + r) / 2;
        if (i <= mid)
            update(o * 2, l, mid, i, val);
        else
            update(o * 2 + 1, mid+1, r, i, val);
        tree[o] = max(tree[o * 2], tree[o*2 + 1]);
    }


    int query(int o, int l, int r, int x) {
        if (r <= x)
            return tree[o];
        int mid = (l + r) / 2;
        if (x <= mid) 
            return query(o * 2, l, mid, x);
        return max(tree[o * 2], query(o * 2 + 1, mid + 1, r, x)); 
    }


    vector<bool> getResults(vector<vector<int>>& queries) {
        int maxn = 0;
        for (auto q : queries)
            maxn = max(maxn, q[1]);
        maxn++;
        
        set<int> st{0, maxn};
        tree.resize(2 << (32 - __builtin_clz(maxn)));
        vector<bool> res;
        
        for (auto q : queries) {
            int x = q[1];
            auto it = st.lower_bound(x);
            int pre = *std::prev(it);
            if (q[0] == 1) {
                int nxt = *it;
                st.insert(x);
                update(1, 0, maxn, x, x - pre);
                update(1, 0, maxn, nxt, nxt - x);
            }
            else {
                int maxlen = max(query(1, 0, maxn, pre), x - pre);
                res.push_back(maxlen >= q[2]);
            }
        }
        return res;
    }
};
相关推荐
海的诗篇_3 分钟前
移除元素-JavaScript【算法学习day.04】
javascript·学习·算法
自动驾驶小卡14 分钟前
A*算法实现原理以及实现步骤(C++)
算法
Unpredictable22215 分钟前
【VINS-Mono算法深度解析:边缘化策略、初始化与关键技术】
c++·笔记·算法·ubuntu·计算机视觉
编程绿豆侠16 分钟前
力扣HOT100之多维动态规划:1143. 最长公共子序列
算法·leetcode·动态规划
珂朵莉MM22 分钟前
2021 RoboCom 世界机器人开发者大赛-高职组(初赛)解题报告 | 珂学家
java·开发语言·人工智能·算法·职场和发展·机器人
fail_to_code1 小时前
递归法的递归函数何时需要返回值
算法
C137的本贾尼1 小时前
(每日一道算法题)二叉树剪枝
算法·机器学习·剪枝
BUG收容所所长3 小时前
栈的奇妙世界:从冰棒到算法的华丽转身
前端·javascript·算法
XRZaaa3 小时前
常见排序算法详解与C语言实现
c语言·算法·排序算法
@我漫长的孤独流浪3 小时前
数据结构测试模拟题(4)
数据结构·c++·算法