AT_abc282_h [ABC282Ex] Min + Sum 题解

按最小值分治。

考虑统计 l,r 内跨过最小值位置 m 的满足条件的区间个数。

遍历 [l,m],[m,r] 中较小的一个区间。设 si=j=1ibj

  • mlrm:遍历左端点 i[l,m],则右端点 j[m,r] 满足 am+x=ijbx=am+sjsi1S,即 sjS+si1am

二分最大的 j0,则 j[m,j0] 都满足条件,答案加上 j0m+1

  • ml>rm:遍历右端点 i[m,r],则左端点 j[l,m] 满足 am+x=jibx=am+sisj1S,即 sj1si+amS

二分最小的 j0,则 j[j0,m] 都满足条件,答案加上 mj0+1

总复杂度 O(nlog2n)

#include <cstdio>
#include <algorithm>
using namespace std;
int n;
long long s, q, a[200050], b[200050], f[20][200050];
long long Q(int l, int r)
{
    int k = __lg(r - l + 1);
    return a[f[k][l]] < a[f[k][r - (1 << k) + 1]] ? f[k][l] : f[k][r - (1 << k) + 1];
}
void F(int l, int r)
{
    if (l > r)
        return;
    int m = Q(l, r);
    if (m - l <= r - m)
        for (int i = l, j; i <= m; ++i)
        {
            j = upper_bound(b + 1, b + r + 1, s + b[i - 1] - a[m]) - b - 1;
            if (j >= m)
                q += j - m + 1;
        }
    else
        for (int i = m, j; i <= r; ++i)
        {
            j = lower_bound(b + l - 1, b + n + 1, b[i] + a[m] - s) - b + 1;
            if (j <= m)
                q += m - j + 1;
        }
    F(l, m - 1);
    F(m + 1, r);
}
int main()
{
    scanf("%d%lld", &n, &s);
    for (int i = 1; i <= n; ++i)
        scanf("%lld", a + i), f[0][i] = i;
    for (int i = 1; 1 << i <= n; ++i)
        for (int j = 1; j + (1 << i) - 1 <= n; ++j)
            f[i][j] = a[f[i - 1][j]] < a[f[i - 1][j + (1 << i - 1)]] ? f[i - 1][j] : f[i - 1][j + (1 << i - 1)];
    for (int i = 1; i <= n; ++i)
        scanf("%d", b + i), b[i] += b[i - 1];
    F(1, n);
    printf("%lld", q);
    return 0;
}
posted @   Jijidawang  阅读(2)  评论(0编辑  收藏  举报  
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具
点击右上角即可分享
微信分享提示