题意:有一个序列,一开始所有的元素都是ai,你可以选择两个长度相等的区间,如果某个元素被一个区间覆盖,那么变为bi,如果被两个区间都覆盖,那么变为ci。问所有区间的选择方法中产生的第k小的元素总和。

首先很容易想到的是二分答案,枚举一个区间,然后用线段树或树状数组查询另一个区间使得元素总和小于等于答案的数量。

为了方便计算,我们把b和c数组中的元素都减掉a,即bi=bi-ai,ci=ci-ai,这样只要查询两区间中元素之和小于等于答案的数量就行了,最后再把答案加上a中元素的和。

由于序列是静态的,因此可以对b和c求个前缀和。

当两区间不相交时,只要把右边的区间从左往右移动,并把左边的区间和动态加入树状数组中查询即可。

当两区间相交时,问题就比较复杂了。按照官方题解中的方法,需要引进两个辅助函数g和f,两者相加会产生"bcb"的效果,这样就又可以用树状数组统计了。

其中$g_x=\sum\limits _{i=1}^{x-1}{(c_i-2b_i)}+\sum\limits _{i=x}^{x+r-1}{(c_i-b_i)}$

$f_y=\sum\limits _{i=1}^{y-1}{(2b_i-c_i)}+\sum\limits _{i=y}^{y+r-1}{b_i}$

推导过程:

设$g_x=\sum\limits _{i=1}^{x-1}{g_1x}+\sum\limits _{i=x}^{x+r-1}{g_2x}$

$f_y=\sum\limits _{i=1}^{y-1}{f_1y}+\sum\limits _{i=y}^{y+r-1}{f_2y}$

则有:$\left\{\begin{matrix}\begin{aligned}&f_1+g_1=0\\&f_1+g_2=b\\&f_2+g_2=c\\&f_2=b\end{aligned}\end{matrix}\right.$

联立方程组即可求出$f_1,f_2,g_1,g_2$的值。

最后要注意的是,由于需要在树状数组上查询的值最多只有O(n)种,因此可以将要查询的值离散化。

 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll N=3e4+,inf=0x3f3f3f3f3f3f3f3fll;
ll n,len,k,a[N],b[N],c[N],C[N*],B[N*],n2,F[N],G[N],K[N],all;
ll lb(ll x) {return x&-x;}
void add(ll u,ll x) {for(; u<=n2; u+=lb(u))C[u]+=x;}
ll get(ll u) {ll ret=; for(; u; u-=lb(u))ret+=C[u]; return ret;}
ll sum(ll* t,ll l,ll r) {return t[r]-t[l-];}
ll g(ll x) {return sum(c,,x-)-*sum(b,,x-)+sum(c,x,x+len-)-sum(b,x,x+len-);}
ll f(ll x) {return *sum(b,,x-)-sum(c,,x-)+sum(b,x,x+len-);}
ll qry(ll x) {
ll cnt=;
for(ll i=; i<=n2; ++i)C[i]=;
for(ll i=len+; i+len-<=n; ++i) {
add(K[i-len]+,);
cnt+=get(upper_bound(B,B+n2,x-B[K[i]])-B);
}
for(ll i=; i<=n2; ++i)C[i]=;
for(ll i=; i+len-<=n; ++i) {
add(G[i-]+,);
if(i-len>=)add(G[i-len]+,-);
cnt+=get(upper_bound(B,B+n2,x-B[F[i]])-B);
}
return cnt;
}
ll bi(ll l,ll r) {
while(l<r) {
ll mid=l+((r-l)>>);
k<=qry(mid)?r=mid:l=mid+;
}
return l;
}
int main() {
scanf("%lld%lld%lld",&n,&len,&k);
for(ll i=; i<=n; ++i)scanf("%lld",&a[i]);
for(ll i=; i<=n; ++i)scanf("%lld",&b[i]);
for(ll i=; i<=n; ++i)scanf("%lld",&c[i]);
for(ll i=; i<=n; ++i)all+=a[i];
for(ll i=; i<=n; ++i)c[i]-=a[i],b[i]-=a[i];
for(ll i=; i<=n; ++i)b[i]+=b[i-],c[i]+=c[i-];
for(ll i=; i+len-<n; ++i)G[i]=g(i),B[n2++]=G[i];
for(ll i=; i+len-<=n; ++i)F[i]=f(i),B[n2++]=F[i];
for(ll i=; i+len-<=n; ++i)K[i]=sum(b,i,i+len-),B[n2++]=K[i];
B[n2++]=~inf,B[n2++]=inf;
sort(B,B+n2),n2=unique(B,B+n2)-B;
for(ll i=; i+len-<n; ++i)G[i]=lower_bound(B,B+n2,G[i])-B;
for(ll i=; i+len-<=n; ++i)F[i]=lower_bound(B,B+n2,F[i])-B;
for(ll i=; i+len-<=n; ++i)K[i]=lower_bound(B,B+n2,K[i])-B;
printf("%lld\n",all+bi(,c[n]));
return ;
}

Gym - 101630G The Great Wall (前缀和+树状数组+二分)的更多相关文章

  1. POJ 2828 Buy Tickets (线段树 or 树状数组+二分)

    题目链接:http://poj.org/problem?id=2828 题意就是给你n个人,然后每个人按顺序插队,问你最终的顺序是怎么样的. 反过来做就很容易了,从最后一个人开始推,最后一个人位置很容 ...

  2. TZOJ 4602 高桥和低桥(二分或树状数组+二分)

    描述 有个脑筋急转弯是这样的:有距离很近的一高一低两座桥,两次洪水之后高桥被淹了两次,低桥却只被淹了一次,为什么?答案是:因为低桥太低了,第一次洪水退去之后水位依然在低桥之上,所以不算“淹了两次”.举 ...

  3. POJ 2182 Lost Cows 【树状数组+二分】

    题目链接:http://poj.org/problem?id=2182 Lost Cows Time Limit: 1000MS   Memory Limit: 65536K Total Submis ...

  4. 树状数组+二分||线段树 HDOJ 5493 Queue

    题目传送门 题意:已知每个人的独一无二的身高以及排在他前面或者后面比他高的人数,问身高字典序最小的排法 分析:首先对身高从矮到高排序,那么可以知道每个人有多少人的身高比他高,那么取较小值(k[i], ...

  5. P2161 [SHOI2009]会场预约[线段树/树状数组+二分/STL]

    题目描述 PP大厦有一间空的礼堂,可以为企业或者单位提供会议场地.这些会议中的大多数都需要连续几天的时间(个别的可能只需要一天),不过场地只有一个,所以不同的会议的时间申请不能够冲突.也就是说,前一个 ...

  6. The Stream of Corning 2( 权值线段树/(树状数组+二分) )

    题意: 有两种操作:1.在[l,r]上插入一条值为val的线段 2.问p位置上值第k小的线段的值(是否存在) 特别的,询问的时候l和p合起来是一个递增序列 1<=l,r<=1e9:1< ...

  7. 牛客多校第3场 J 思维+树状数组+二分

    牛客多校第3场 J 思维+树状数组+二分 传送门:https://ac.nowcoder.com/acm/contest/883/J 题意: 给你q个询问,和一个队列容量f 询问有两种操作: 0.访问 ...

  8. BZOJ-2743: [HEOI2012]采花 前缀和 树状数组

    BZOJ-2743 LUOGU:https://www.luogu.org/problemnew/show/P4113 题意: 给一个n长度的序列,m次询问区间,问区间中出现两次及以上的数字的个数.n ...

  9. [CSP-S模拟测试]:斯诺(snow)(数学+前缀和+树状数组)

    题目传送门(内部题37) 输入格式 第一行一个整数$n$,表示区间的长度. 第二行一个长度为$n$的只包含$0,1,2$的字符串,表示给出的序列. 输出格式 一行一个整数,表示革命的区间的数量. 样例 ...

随机推荐

  1. 英特尔® 图形性能分析器 2019 R1 版本

    了解并下载全新英特尔® 图形性能分析器 2019 R1 版本.新版本新增了 DX11 和 Vulkan 多帧流捕获模式,可以在“帧和图形跟踪分析器”中分析 Vulkan 应用.此外,帧分析器还添加了 ...

  2. 什么是java的线程安全?同步,异步

    线程是比进程更小的执行单位,是在进程基础上进行的进一步划分.所谓多线程是指进程在执行过程中可以产生多个同时存在.同时运行的线程.多进程机制可以合理利用资源,提高程序的运行效率.一个进程至少包含一个线程 ...

  3. jenkins自动化部署springboot

    一.linux按jar包名称部署 1.执行shell PID=$(ps -ef | grep app.jar | grep -v grep | awk '{ print $2 }') if [ -z ...

  4. CDH的mysql主从准备

    参考: https://www.cnblogs.com/yinzhengjie/p/10371899.html https://www.sysit.cn/blog/post/sysit/CDH6.2. ...

  5. 06: zabbix常见面试题

    1.1 zabbix架构 1.zabbix理论 1)Zabbix是一个企业级的.开源的.分布式的监控套件,Zabbix可以监控网络和服务的监控状况. 2)Zabbix利用灵活的告警机制,允许用户对事件 ...

  6. Hive 教程(五)-参数配置

    配置基本操作 hive> set; 查看所有配置hive> set key: 查看某个配置hive> set key value: 设置某个配置 我们可以看到一些 hadoop 的配 ...

  7. shell脚本获取的参数

    $# 是传给脚本的参数个数 $0 是脚本本身的名字 $1 是传递给该shell脚本的第一个参数 $2 是传递给该shell脚本的第二个参数 $@ 是传给脚本的所有参数的列表

  8. command----常用命令更新ing

    common commands 1:split---拆分文件 [root@localhost split]# split -b 1M split.tar.gz split_ #按1M拆分文件 [roo ...

  9. java代码--Date类获取当前时间-格式化输出

    44:52 阅读数:2299 package cn.Date;   import java.text.Format; import java.text.SimpleDateFormat; import ...

  10. luogu题解 UVA11536 【Smallest Sub-Array】最短set区间&滑动窗口

    题目链接: https://www.luogu.org/problemnew/show/UVA11536 题目大意: 给定一个\(N,M,K\),构造这样的数列: \(x[1]=1,x[2]=2,x[ ...