首页 > 其他分享 >AT1330 题解

AT1330 题解

时间:2022-08-26 02:00:57浏览次数:124  
标签:二分 log int 题解 LL AT1330 mid 答案

前言

题目传送门!

更好的阅读体验?

这一题内部比赛时考到了,个人觉得是一道二分答案好题。

本题时间很宽松,导致 \(O(n \log^2 n)\) 的代码可以跑过去。

但是,我内部比赛的时限是 \(1\) 秒,这就导致需要 \(O(n \log n)\) 的代码了。

思路一

显然是一道二分答案题目。

二分答案老套路,设 \(f(x)\) 表示 \(x\) 是否能作为答案,易得 \(f(x)\) 单调递增。

所以,可以使用二分答案。

重点是 \(\texttt{check()}\) 函数如何编写,我们可以使用贪心的思想。

对于每个气球,我们可以得出,打掉它的时间 \(t_i\) 不得超过 \(\left\lfloor\dfrac{x - h_i}{s_i}\right\rfloor\)。

我们可以对 \(t\) 数组从小到大排序。

对于 \(0 \le i < n\),我们从贪心的角度思考。如果 \(i > t_i\),说明已经无法满足 \(x\) 了。

如果全部都符合,说明 \(t\) 数组的攻击方式就是一种合法的方案,那么 \(x\) 这个答案是可行的。

\(\texttt{check()}\) 函数代码如下。

typedef long long LL;
int n, h[N], s[N];
LL t[N];
bool chk(LL x)
{
    for (int i = 1; i <= n; i++)
    {
        //如果时刻 0 打掉它都无法满足,立刻叉掉。
        if (x < h[i]) return false;
        t[i-1] = (x - h[i]) / s[i]; //为了方便处理,下标从 0 开始。
    }
    sort(t, t+n);
    for (int i = 0; i < n; i++)
        if (t[i] < i)
            return false;
    return true;
}

这里的时间复杂度是 \(O(n \log n)\),加上二分的板子,需要 \(O(n \log^2 n)\)。

可以通过此题,但还可以优化吗?

思路二

时间复杂度瓶颈在于排序,如果想降到 \(O(n)\),就需要使用 \(O(n)\) 的排序算法。

你想到什么方法了?对,桶排序

我们发现,当 \(t_i \ge n\),说明任意时刻打掉它都是可行的,所以可以忽略不计。

这样,桶数组空间就符合了。

需要注意,\(\texttt{check()}\) 开始前要初始化桶。

bool chk(LL x)
{
    memset(box, 0, sizeof(box));
    for (int i = 1; i <= n; i++)
    {
        //在主程序外面保证了 x 大于等于 h[i],就不需要担心了。
        LL t = (x - h[i]) / s[i];
        if (t < n) box[t]++;
    }
    int cur = 0;
    for (int i = 0; i < n; i++)
        for (int j = 1; j <= box[i]; j++)
        {
            if (i < cur) return false;
            cur++;
        }
    return true;
}

时间复杂度终于优化成了 \(O(n)\)。

最后,我们补全二分。

LL FIND(LL l, LL r)
{
    //显然是模版,完全没改变。
    while (l < r)
    {
        LL mid = l + r >> 1;
        if (chk(mid)) r = mid;
        else l = mid+1;
    }
    return r;
}
int main()
{
    scanf("%d", &n);
    int maxn = -1;
    for (int i = 1; i <= n; i++) scanf("%d%d", &h[i], &s[i]), maxn = max(maxn, h[i]);
    //从 maxn 开始,保证了 chk() 函数不会出现 x < h[i] 的情况。
    //10^9 + 10^5 * 10^9 近似看成 10^15,毕竟大一点没坏处。
    printf("%lld\n", FIND(maxn, 1e15));
    return 0;
}

希望对大家有帮助!
首发:2022-07-02 10:33:00

标签:二分,log,int,题解,LL,AT1330,mid,答案
From: https://www.cnblogs.com/liangbowen/p/16622894.html

相关文章

  • P2130 题解
    前言题目传送门!更好的阅读体验?本题是练习bfs的好题。思路结合代码进行思路讲解。首先是读入部分,我们可以用bool存下地图,节省空间开销。需要注意,数据比较烂,起始......
  • CF1635B 题解
    题目传送门!更好的阅读体验?这题显然可以使用贪心的思想解决。由于头和尾一定不用更改,所以只需从\(a_2\)枚举到\(a_{n-1}\)。贪心原则下,我们更改的数应该要与相邻的......
  • P8295 题解
    ###前言题目传送门\(\color{red}{see}\space\color{green}{in}\space\color{blue}{my}\space\color{purple}{blog}\)这题并不困难,代码也挺短的,题目理解稍有困难。......
  • 题解 UVA10869 Brownie Points II
    题意平面上有若干点,\(\text{stan}\)通过一个点划了一条直线,\(\text{ollie}\)通过在这条直线上的点作了一条垂线,那么平面被分成\(4\)个象限,\(\text{stan}\)获得的分数......
  • idea导入依赖maven的dependenci列表报红问题解决
    打开一个idea的pom文件时,明明仓库有相关依赖,并且maven的仓库配置没有错误,但是maven的dependencies列表却报红,我们可以让idea每次加载pom文件的依赖不从idea的缓存中读取,而......
  • npm 报错:npm ERR! Maximum call stack size exceeded 超过最大栈问题解决方案
       错误的原因,npm版本问题;解决办法:   1、更新到最新版本:npminstallnpm-g  要记住全局更新2、回退版本:npminstall-gnpm@5.4.0 ......
  • 蔚来杯2022牛客暑期多校训练营10 题解
    D.MiReDoSiLa?SoFa![NOI2016]优秀的拆分原题。枚举周期\(k\),并将位置为\(k\)的倍数的点设为关键点。枚举相邻两个点\(i,i+k\),并求出\(lcp(S[i...n],S[i+k......
  • 2022“杭电杯”中国大学生算法设计超级联赛(10) 题解
    C.WavyTree发现修改次数和相邻两数的相对大小有关,所以可先求出差分数组。分两种情况考虑:①奇数位置为波峰②偶数位置为波峰。以情况①为例,若奇数位置差分后值小......
  • Idea创建Maven Web工程的web.xml版本问题解决
    问题使用Maven创建web工程的时候,创建出来的web.xml版本有问题。临时解决方案将在Tomcat安装目录下的webapps/ROOT/WEB-INF下的web.xml替换项目下的web.xml......
  • jmeter-特殊问题解决
    1、相应报文乱码问题:方法一:1、在相应节点的下方,比如http请求,添加后置处理器–》BeanShellPostProcessor2、然后在其脚本框中添加如下代码prev.setDataEncoding(“UTF-8......