当前位置:首页 > 技术 > 正文内容

线段树上的二分查找优化技术

访客 技术 2026年5月26日 4

线段树上二分:从 O(log²n) 到 O(log n)

在处理区间查询问题时,我们常遇到这样的需求:在给定区间内快速定位第一个(或最后一个)满足某种条件的位置。例如,在区间 [l, r] 中找出第一个值大于 x 的元素下标。朴素做法是结合二分查找与线段树查询,但每次 check 操作需要 O(log n) 时间,总复杂度为 O(log²n)。 本文介绍一种更高效的技巧——**在线段树结构内部直接进行二分决策**,将时间复杂度优化至 O(log n)。

核心思想

传统方法是在外层二分位置,并对每个候选位置调用线段树查询其对应区间的最大值/最小值等信息。而本优化的关键在于:**利用线段树本身的结构,在递归过程中根据左右子树的信息直接决定走向哪一侧**,避免重复查询。 具体来说: - 若当前节点维护的区间完全包含于目标查询区间,则可直接依据左、右子树的最大值判断下一步方向; - 否则按标准线段树遍历方式进入与其相交的子区间; - 在每一步都提前判断是否存在可行解,若整个子树都不满足条件则立即返回 -1。

基础版本:全局首个大于 x 的位置

假设我们要在 [1, n] 范围内找到第一个值大于 x 的下标:
int find_first(int u, int x) {
    // 整个区间的最大值都不超过x,无解
    if (tree[u].max_val <= x) return -1;

    // 叶子节点,说明找到了答案
    if (tree[u].l == tree[u].r) return tree[u].l;

    // 根据左子树是否存在解来决定走向
    if (tree[u * 2].max_val > x)
        return find_first(u * 2, x);
    else
        return find_first(u * 2 + 1, x);
}
该过程仅沿一条路径向下,故时间复杂度为 O(log n)。

扩展到任意区间 [l, r]

当查询范围受限于 [L, R] 时,需区分三种情况:
  1. 当前节点区间完全被 [L, R] 包含 → 可以在此节点上做二分决策;
  2. 部分重叠 → 继续递归到与 [L, R] 相交的子节点;
  3. 无交集 → 忽略。
实现如下:
int query_range(int u, int L, int R, int x) {
    // 提前剪枝:本子树无有效解
    if (tree[u].max_val <= x) return -1;

    if (tree[u].l == tree[u].r)
        return tree[u].l;

    int mid = (tree[u].l + tree[u].r) >> 1;
    
    // 完全包含于查询区间:允许内部二分
    if (L <= tree[u].l && tree[u].r <= R) {
        if (tree[u * 2].max_val > x)
            return query_range(u * 2, L, R, x);
        else
            return query_range(u * 2 + 1, L, R, x);
    }
    // 部分相交:分别尝试左右子树
    else {
        int res = -1;
        if (L <= mid)
            res = query_range(u * 2, L, R, x);
        if (res == -1 && R > mid)
            res = query_range(u * 2 + 1, L, R, x);
        return res;
    }
}
注意:**必须在函数入口处判断 max_val ≤ x 的情况**,否则可能导致多个无效子树被完整遍历,使最坏复杂度退化为 O(log²n)。

寻找最右侧满足条件的位置

只需调整优先访问顺序,让系统尽量往右走即可:
int query_rightmost(int u, int L, R, int x) {
    if (tree[u].max_val <= x) return -1;

    if (tree[u].l == tree[u].r)
        return tree[u].l;

    int mid = (tree[u].l + tree[u].r) >> 1;

    if (L <= tree[u].l && tree[u].r <= R) {
        // 优先检查右子树
        if (tree[u * 2 + 1].max_val > x)
            return query_rightmost(u * 2 + 1, L, R, x);
        else
            return query_rightmost(u * 2, L, R, x);
    } else {
        int res = -1;
        if (R > mid)
            res = query_rightmost(u * 2 + 1, L, R, x);
        if (res == -1 && L <= mid)
            res = query_leftmost(u * 2, L, R, x);
        return res;
    }
}

注意事项

  • 涉及懒更新时,务必在进入子节点前执行 pushdown 操作;
  • 所有递归入口均应包含有效性判断,防止无效分支展开;
  • 此方法适用于支持区间合并性质的问题,如最大值、最小值、和等单调性信息。

典型应用题单

相关文章

Linux crontab 详解

1) crontab 是什么cron 是 Linux 的定时任务守护进程;crontab 是用来编辑/查看“按时间周期执行命令”的表(cron table)。常见两类:用户 crontab:每个用户一份(crontab -e 编辑)系统级 crontab / cron.d:可指定执行用户(/etc/crontab、/etc/cron.d/*)2) crontab 时间...

富文本里可以允许的 HTML 属性

一、所有标签默认允许的安全属性(极少)class        (可选)id           (通常建议禁用)title️ 注意:id 容易被滥用做锚点注入,很多系统直接禁用class 允许的话最好只允许固定前缀(如 editor-*)二、a 标签允许属性<a href="" t...

Mac 安装 Node.js 指南

方法一:通过官网安装包(最简单,适合初学者)如果你只是想快速安装并开始使用,这是最直接的方法。访问 Node.js 官网。页面会显示两个版本:LTS (Recommended For Most Users):长期支持版,最稳定。建议选这个。Current:最新特性版,包含最新功能但可能不够稳定。下载 .pkg 安装包并运行。按照安装向导点击“下一步”即可完成。方法二:使用 Homebrew 安装(...

Dom\HTML_NO_DEFAULT_NS 的副作用:自动加闭合标签

在使用Dom\HTMLDocument时,Dom\HTML_NO_DEFAULT_NS 将禁止在解析过程中设置元素的命名空间, 此设置是为了与DOMDocument向后兼容而存在的。当使用它时,已知的一个副作用就是:自动加闭合标签例如 </img> 为什么会这样?当你使用:Dom\HTML_NO_DEFAULT_NS文档会变成 无命名空间模式,此时内部更接近 XML...

Laravel 事件和监听器创建

在 Laravel 中,使用 Artisan 命令创建 Events(事件) 和 Listeners(监听器) 是非常高效的。你可以通过以下几种方式来实现:1. 手动创建单个 Event如果你只想创建一个事件类,可以使用 make:event 命令:Bashphp artisan make:event UserRegistered执行后,文件将生成在 app/Even...

自定义域名解析神器 dnsmasq

什么是 dnsmasq?dnsmasq 是一个轻量级、功能强大的网络服务工具,专为小型和中等规模网络设计。它是一个综合的网络基础设施解决方案[1]。dnsmasq 能做什么?功能说明应用场景DNS 转发与缓存将 DNS 查询转发到上游服务器(ISP、Google DNS 等),并在本地缓存结果加快 DNS 查询速度,减少外部 DNS 流量本地 DNS解析本地网络设备的主机名,无需编辑&n...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。