正题

题目链接:https://www.ybtoj.com.cn/problem/643


题目大意

\(n\)个机器人,第\(i\)个攻击力为\(A_i\),防御为\(D_i\)。

然后你每次可以对一个机器人造成\(Atk\)点伤害,之后所有机器人对你进行一次攻击。

开局可以删除两个机器人,求最少受到多少伤害。

\(n\in[3,3\times 10^5],A_i,T_i\in[1,10^4]\)


解题思路

设每个机器人需要攻击的次数\(T_i\)

先不考虑删除的话是一个很经典的贪心,按照\(\frac{T_i}{A_i}\)从小到大排序就好了。证明的话

设目前是排序好的序列,是否交换相邻的两个\(i,j(j>i)\)需要满足

\[T_iA_j\geq T_jA_i
\]

化简一下就可以发现一定不合法

然后考虑删除哪两个,设\(St_i=\sum_{j=1}^iT_i,Sa_i=\sum_{j=1}^nA_i\),那么删除一个\(x\)会减少贡献

\[b_x=(Sa_n-Sa_x)T_x+St_xA_x-A_x
\]

(分别计算自己减去的和自己对后面的数产生的贡献)。

但是如果删除了两个数\(x,y(x<y)\)就会多减去\(T_xA_y\)的贡献。

所以我们要求$$\max{ b_x+b_y-T_xA_y } (x<y)$$

这个因为值域比较小直接上李超树就好了,当然也可以\(CDQ\)分治或者\(Splay\)搞斜率优化

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


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const ll N=3e5+10;
struct node{
ll b,k;
};
ll n,atk,sum,ans,maxs,lim;
ll p[N],a[N],t[N],st[N],sa[N],b[N],w[N];
bool cmp(ll x,ll y)
{return t[x]*a[y]<t[y]*a[x];}
ll ct(ll x,ll id)
{return t[p[id]]*x+b[id];}
void Change(ll x,ll l,ll r,ll id){
if(ct(l,id)>=ct(l,w[x])&&ct(r,id)>=ct(l,w[x])){w[x]=id;return;}
if(ct(l,id)<=ct(l,w[x])&&ct(r,id)<=ct(r,w[x]))return;
if(l==r)return;ll mid=(l+r)>>1;
if(t[p[id]]<t[p[w[x]]]){
if(ct(mid,id)>=ct(mid,w[x]))
Change(x*2+1,mid+1,r,w[x]),w[x]=id;
Change(x*2,l,mid,id);
}
else{
if(ct(mid,id)>=ct(mid,w[x]))
Change(x*2,l,mid,w[x]),w[x]=id;
Change(x*2+1,mid+1,r,id);
}
return;
}
ll Ask(ll x,ll l,ll r,ll pos){
if(l==r)return ct(pos,w[x]);
ll mid=(l+r)>>1,ans;
if(pos<=mid)ans=Ask(x*2,l,mid,pos);
else ans=Ask(x*2+1,mid+1,r,pos);
return max(ans,ct(pos,w[x]));
}
signed main()
{
freopen("fittest.in","r",stdin);
freopen("fittest.out","w",stdout);
scanf("%lld%lld",&n,&atk);
for(ll i=1;i<=n;i++){
scanf("%lld%lld",&a[i],&t[i]);
t[i]=(t[i]+atk-1)/atk;p[i]=i;
sum+=a[i];lim=max(max(a[i],t[i]),lim);
}
sort(p+1,p+1+n,cmp);b[0]=-1e18;
for(ll i=1;i<=n;i++){
ll x=p[i];
st[i]=st[i-1]+t[x];
sa[i]=sa[i-1]+a[x];
b[i]=(sum-sa[i])*t[x]+st[i]*a[x]-a[x];
ans+=st[i]*a[x]-a[x];
}
for(ll i=1;i<=n;i++){
int x=p[i];
ll tmp=b[i]+Ask(1,1,lim,a[x]);
maxs=max(maxs,tmp);t[x]=-t[x];
Change(1,1,lim,i);
}
printf("%lld\n",ans-maxs);
return 0;
}

YbtOJ#643-机器决斗【贪心,李超树】的更多相关文章

  1. BZOJ4391 High Card Low Card [Usaco2015 dec](贪心+线段树/set库

    正解:贪心+线段树/set库 解题报告: 算辣直接甩链接qwq 恩这题就贪心?从前往后从后往前各推一次然后找一遍哪个地方最大就欧克了,正确性很容易证明 (这里有个,很妙的想法,就是,从后往前推从前往后 ...

  2. BZOJ1568: [JSOI2008]Blue Mary开公司【李超树】

    Description Input 第一行 :一个整数N ,表示方案和询问的总数. 接下来N行,每行开头一个单词"Query"或"Project". 若单词为Q ...

  3. #6034. 「雅礼集训 2017 Day2」线段游戏 李超树

    #6034. 「雅礼集训 2017 Day2」线段游戏 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统 ...

  4. Good Inflation SPOJ - GOODG 李超树

    题目传送门 题意:刚开始有一个气球体积为空,现在有n个充气点,从1->n遍历这n充气点,每个充气点有vi,di,vi为走到这个充气点之后可以为气球充气vi的体积,di为选择了在这个点充气的时候, ...

  5. 【题解】P1712 [NOI2016]区间(贪心+线段树)

    [题解]P1712 [NOI2016]区间(贪心+线段树) 一个observe是,对于一个合法的方案,将其线段长度按照从大到小排序后,他极差的来源是第一个和最后一个.或者说,读入的线段按照长度分类后, ...

  6. [SDOI2016]游戏(树剖+李超树)

    趁着我把李超树忘个一干二净的时候来复习一下吧,毕竟马上NOI了. 题解:看着那个dis就很不爽,直接把它转换成深度问题,然后一条直线x->y,假设其lca为z,可以拆分成x->z和z-&g ...

  7. P4254 [JSOI2008]Blue Mary开公司 (李超树)

    题意:插入一些一次函数线段 每次询问在x = x0处这些线段的最大值 题解:李超树模版题 维护优势线段 注意这题的输入是x=1时的b #include <iostream> #includ ...

  8. Codeforces 675E Trains and Statistic(DP + 贪心 + 线段树)

    题目大概说有n(<=10W)个车站,每个车站i卖到车站i+1...a[i]的票,p[i][j]表示从车站i到车站j所需买的最少车票数,求所有的p[i][j](i<j)的和. 好难,不会写. ...

  9. poj 2010 Moo University - Financial Aid (贪心+线段树)

    转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlove 骗一下访问量.... 题意大概是:从c个中选出n个 ...

随机推荐

  1. 如何快速排查发现redis的bigkey?4种方案一次性给到你!

    本篇文章将以redis的bigkey为主题进行技术展开,通过从认识redis的高性能,bigkey的危害.存在原因.4种解决方案,到模拟实战演练的介绍方式,来跟大家一起认识.探讨和学习redis. 先 ...

  2. mysql 数据库 分表后 怎么进行分页查询?Mysql分库分表方案?

    Mysql分库分表方案 1.为什么要分表: 当一张表的数据达到几千万时,你查询一次所花的时间会变多,如果有联合查询的话,我想有可能会死在那儿了.分表的目的就在于此,减小数据库的负担,缩短查询时间. m ...

  3. Servlet过滤器----Filter

    JavaEE的Servlet规范描述了三种技术:Servlet,Filter,Listener (一)过滤器简介 Filter也称之为过滤器,它是Servlet技术中最实用的技术,WEB开发人员通过F ...

  4. MPI集群搭建

    高性能计算     ubantu下集群搭建 参考博客:https://blog.csdn.net/u012304016/article/details/52423738(尊重别人的知识产权),一些细节 ...

  5. python使用UTF-8写入CSV中文乱码

    使用encoding='utf-8',写入的文档是乱码. 解决办法: 修改encoding='utf-8-sig' 关于文件open()函数: open(path,'-模式-',encoding='u ...

  6. HZ游记

    HZ 游记 Day -1 收拾东西,准备出发. 话说这几天一直比较懒,也没什么心情和效率学习,颓废好几天了,希望到衡水以后能感觉好点. 不知道衡水有没有妹子 非常想看看衡水的样子,但是又害怕封闭式教学 ...

  7. Git 学习路线

    前言 感觉 Git 还是很重要,应该单独开一篇文章来讲 Git... 使用系列教程 Git 系列教程(1)- Git 简介 Git 系列教程(2)- Git 安装 Git 系列教程(3)- 初次运行 ...

  8. Linux - 解决使用 apt-get 安装 yum 的时耗报 E: Unable to locate package yum 的错误

    问题背景 在 Linux 系统下使用 apt-get 命令安装 yum 库报错 apt-get install yum E: Unable to locate package yum 问题解决 一行命 ...

  9. 通过JDK动态代理实现 Spring AOP

    1.新建一个目标类 接口:public interface IUserService //切面编程 public void addUser(); public void updateUser( ); ...

  10. epoll经典代码示例

    1. epoll原理 原理性的知识不再另做说明,我在这里附上收藏整理的两篇经典文章: select与epoll的本质关系. select.poll.epoll之间的区别. 2. epoll服务器端经典 ...