Educational CodeForces Round 168 E Solution Optimization

闲话

笑点解析:砍一个 \(\log\) 省下十几毫秒,上基础版快读省下 \(250\) 毫秒。

题解

在讲解优化方法前请确保自己已经理解了这篇题解

根据上述的题解,我们需要以更快的方法找到使得每个怪兽 \(i\) 坚持应战的阈值 \(c_i\)

优化 \(1\) - 树状数组内部操作

众所周知,query 函数是这么写的:

inline int query(int x)
{
    int res = 0;
    while (x)
        res += tr[x], x -= (x & -x);
    return res;
}

换句话说,query 函数干的事就是一个一个削掉 \(x\)lowbit,顺便加上对应位置上的 \(\text{tr}_x\)

也就是说,如果我们给 \(x\) 加上一个小于它 lowbit 的数 \(y\),那么 \(\text{query}(x+y)=\text{query}(x)+\text{tr}_{x+y}\)

利用这个性质,我们可以从上向下枚举 \(c_i\) 的每个二进制位,若怪兽会应战,就忽略;否则,加上这个二进制位。

for (int i = 1; i <= n; i++)
{
    l = cur = 0;
    for (int j = 17; ~j; j--)
    {
        if (1ll * a[i] * (l | (1 << j)) <= cur + tr[l | (1 << j)])
            l |= (1 << j), cur += tr[l];
    }
    l++;
    update(l, 1);
    req[i] = l;
}

时间复杂度 \(O(n\log \max(a))\)

优化 \(2\) - 输入优化

scanfgetchar 慢是一个老生常谈的道理。

template <typename _Tp>
inline void read(_Tp &x)
{
    char ch;
    while (ch = getchar(), !isdigit(ch))
        ;
    x = ch - '0';
    while (ch = getchar(), isdigit(ch))
        x = (x << 3) + (x << 1) + (ch ^ '0');
}
template <typename _Tp, typename... _Args>
inline void read(_Tp &x, _Args &...args)
{
    read(x);
    read(args...);
}

支持可变参数调用,代码是从之前一次洛谷月赛那里学的,但是忘了是哪一场。

代码

最终运行时间 \(187\) 毫秒。评测记录在这里

#include <iostream>
#include <tuple>
using namespace std;
const int N = 4e5 + 10;
int a[N], n, q, tr[N], idx = 1, l, r, mid, req[N], cur;
template <typename _Tp>
inline void read(_Tp &x)
{
    char ch;
    while (ch = getchar(), !isdigit(ch))
        ;
    x = ch - '0';
    while (ch = getchar(), isdigit(ch))
        x = (x << 3) + (x << 1) + (ch ^ '0');
}
template <typename _Tp, typename... _Args>
inline void read(_Tp &x, _Args &...args)
{
    read(x);
    read(args...);
}
inline void update(int x, int v)
{
    while (x < N)
    {
        tr[x] += v;
        x += (x & -x);
    }
}
int main()
{
    read(n, q);
    for (int i = 1; i <= n; i++)
    {
        read(a[i]);
    }
    for (int i = 1; i <= n; i++)
    {
        l = cur = 0;
        for (int j = 17; ~j; j--)
        {
            if (1ll * a[i] * (l | (1 << j)) <= cur + tr[l | (1 << j)])
                l |= (1 << j), cur += tr[l];
        }
        l++;
        update(l, 1);
        req[i] = l;
    }
    for (int i = 1, x, k; i <= q; i++)
    {
        read(x, k);
        puts(k < req[x] ? "NO" : "YES");
    }
}
posted @ 2024-08-01 23:59  丝羽绫华  阅读(12)  评论(0编辑  收藏  举报