CF1039E Summer Oenothera Exhibition
根号分治好题。
可以先看我的根号分治总结。
题意就是给出长度为\(n\)的区间和\(q\)组询问以及一个\(w\),每次询问一个\(k\),问最少把一段给定区间划分几次可以满足每一段划分出的子区间的极差不超过\(w-k\)(以下默认\(k\)就是\(w-k\))。
这题主要有两种写法,一种是\(O(n \sqrt nlog n)\)的,一种是\(O(n^{ \frac 5 3}+n^{ \frac 4 3} log n)\)的(本文中默认了\(n\)和\(q\)同阶)。
反正先考虑\(n^2\)暴力,预处理一个ST表,对于每次询问\(O(n)\)地贪心扫一遍,能划在一个子区间内就划在一个子区间内,这样贪心一定是正确的。
先讲第一种,比较好想,想到从每个点向从它开始能划到它同一个子区间内的最远的点的右边一个点(定义这个点为\(h[j]\))一条边,这时我们就想起了弹飞绵羊,可以用LCT维护答案,但是由于每次改变\(k\)值会可能更改所有点连边情况,的直接这样做是\(O(n^2 log n)\)的,连暴力都不如。考虑根号分块,先把询问离线,按\(k\)从小到大排个序,这样划分出区间的长度是单调不减的,如果所连的边的长度不超过\(\sqrt n\),就直接用LCT暴力维护,由于边的长度不超过\(\sqrt n\),对于所有点最多删边加边\(\sqrt n\)次,由于每次的复杂度都是\(log n\)的,所以总的复杂度是\(O(n \sqrt n log n)\);如果所要连的边长度超过\(\sqrt n\)了,就不连这条边;这样我们就用LCT维护了一个森林,对于森林里每棵树的贡献都可以\(O(1)\)查询,每次询问时先计算一棵树的贡献,到了树根之后就二分它的\(h[j]\),跳过去把答案++,再计算\(h[j]\)所在树的贡献。
#include<cstdio>
#include<cctype>
#include<algorithm>
#include<vector>
#include<cmath>
#define R register
#define I inline
using namespace std;
const int S=100003,M=17,inf=0x7fffffff;
char buf[1000000],*p1,*p2;
I char gc(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,S,stdin),p1==p2)?EOF:*p1++;}
I int rd(){
R int f=0; R char c=gc();
while(c<48||c>57) c=gc();
while(c>47&&c<58) f=f*10+(c^48),c=gc();
return f;
}
struct query{int k,p,o;}q[S];
struct splay{int p,d[2],s;}e[S];
int a[S],b[S],l[S],f[S][M],g[S][M],h[S],m;
vector<int> v[S];
I int operator<(query x,query y){return x.k<y.k;}
I int min(int x,int y){return x<y?x:y;}
I int max(int x,int y){return x>y?x:y;}
I int qmn(int x,int y){
R int i=l[y-x+1];
return min(f[x][i],f[y-(1<<i)+1][i]);
}
I int qmx(int x,int y){
R int i=l[y-x+1];
return max(g[x][i],g[y-(1<<i)+1][i]);
}
I int nrt(int x){return e[e[x].p].d[0]==x||e[e[x].p].d[1]==x;}
I void psu(int x){e[x].s=e[e[x].d[0]].s+e[e[x].d[1]].s+1;}
I void rtt(int x){
R int f=e[x].p,g=e[f].p,b=e[f].d[1]==x,c=e[x].d[!b];
if(nrt(f)) e[g].d[e[g].d[1]==f]=x;
if(c) e[c].p=f;
e[f].p=x,e[x].p=g,e[x].d[!b]=f,e[f].d[b]=c,psu(f);
}
I void spl(int x){
for(R int f,g;nrt(x);rtt(x)){
f=e[x].p,g=e[f].p;
if(nrt(f))
rtt((e[g].d[1]==f)^(e[f].d[1]==x)?x:f);
}
psu(x);
}
I void acc(int x){
for(R int y=0;x;x=e[y=x].p)
spl(x),e[x].d[1]=y,psu(x);
}
I int frt(int x){
while(e[x].d[0])
x=e[x].d[0];
return x;
}
I void cut(int x){acc(x),spl(x),e[e[x].d[0]].p=0,e[x].d[0]=0,psu(x);}
I int fnd(int x,int k){
R int l=x+1,r=m,m;
for(m=l+r>>1;l<r;m=l+r>>1)
qmx(x,m)-qmn(x,m)>k?r=m:l=m+1;
return m;
}
int main()
R int n=rd(),W=rd(),Q=rd(),p=sqrt(n),i,j,k,t,s,o;
for(m=n+1,i=1;i<=n;++i)
a[i]=f[i][0]=g[i][0]=rd();
a[m]=f[m][0]=g[m][0]=inf,l[1]=0;
for(i=2;i<=m;++i)
l[i]=l[i>>1]+1;
for(i=1,k=2;k<=m;++i,k<<=1)
for(j=1;j+k-1<=m;++j)
f[j][i]=min(f[j][i-1],f[j+(k>>1)][i-1]),g[j][i]=max(g[j][i-1],g[j+(k>>1)][i-1]);
for(i=1;i<=Q;++i)
q[i].p=i,q[i].k=W-rd();
sort(q+1,q+Q+1);
for(i=1;i<=n;++i)
h[i]=i,e[i].p=i+1,v[1].push_back(i);
for(i=1;i<=Q;++i){
for(o=0,t=0,s=v[i].size();t<s;++t){
j=v[i][t],cut(j);
for(k=h[j]+1;qmx(j,k)-qmn(j,k)<=q[i].k&&k-j<=p&&k<=n;++k);
if(k-j>p)
b[j]=1;
else
h[j]=k,e[j].p=h[j],v[lower_bound(q+1,q+1+Q,(query){qmx(j,h[j])-qmn(j,h[j]),0})-q].push_back(j);
}
for(j=1;;j=fnd(j,q[i].k),++o){
if(!b[j])
acc(j),spl(j),o+=e[j].s-1,j=frt(j);
if(j>n)
break;
}
q[q[i].p].o=o-1;
}
for(i=1;i<=Q;++i)
printf("%d\n",q[i].o);
return 0;
}
注意在实现时有一个重要的细节,删边加边的时候必须二分找它下一个可能改变它\(h[j]\)值的询问,而不能对于每次询问都把序列扫一遍,这样复杂度会被卡成\(n^2\)。例如下面的操作就是错误的。
for(i=1;i<=Q;++i){
o=0;
for(j=1;j<=n;++j)
if(!b[j]){
for(k=h[j];qmx(j,k)-qmn(j,k)<=q[i].k&&k-j<=p&&k<=n;++k);
if(k-j>p)
b[j]=1,cut(j);
else
if(k^h[j])
cut(j),h[j]=k,e[j].p=h[j];
}
for(j=1;;j=fnd(j,q[i].k),++o){
if(!b[j])
acc(j),spl(j),o+=e[j].s-1,j=frt(j);
if(j>n) break;
}
q[q[i].p].o=o-1;
}
下面介绍第二种做法。
wxh太神了我看不懂。
然后Itst把wxh的做法写出来了,我就直接丢链接了。
CF1039E Summer Oenothera Exhibition 根号分治,LCT,ST表的更多相关文章
-
[CF1039E]Summer Oenothera Exhibition[根号分治+lct]
题意 给一个长度为 \(n\) 的序列, \(q\) 次询问,次给一个 \(k_i\) ,问最少将序列划分成多少次,满足每一段的极差不超过\(w−k_i\). \(1 \leq n, q \leq 1 ...
-
CF1039E Summer Oenothera Exhibition 贪心、根号分治、倍增、ST表
传送门 感谢这一篇博客的指导(Orzwxh) $PS$:默认数组下标为$1$到$N$ 首先很明显的贪心:每一次都选择尽可能长的区间 不妨设$d_i$表示在取当前$K$的情况下,左端点为$i$的所有满足 ...
-
【BZOJ3784】树上的路径 点分治序+ST表
[BZOJ3784]树上的路径 Description 给定一个N个结点的树,结点用正整数1..N编号.每条边有一个正整数权值.用d(a,b)表示从结点a到结点b路边上经过边的权值.其中要求a< ...
-
CF587F-Duff is Mad【AC自动机,根号分治】
正题 题目链接:https://www.luogu.com.cn/problem/CF587F 题目大意 给出\(n\)个字符串\(s\).\(q\)次询问给出\(l,r,k\)要求输出\(s_{l. ...
-
Codeforces 1039D You Are Given a Tree [根号分治,整体二分,贪心]
洛谷 Codeforces 根号分治真是妙啊. 思路 考虑对于单独的一个\(k\)如何计算答案. 与"赛道修建"非常相似,但那题要求边,这题要求点,所以更加简单. 在每一个点贪心地 ...
-
BZOJ.4320.[ShangHai2006]Homework(根号分治 分块)
BZOJ \(\mathbb{mod}\)一个数\(y\)的最小值,可以考虑枚举剩余系,也就是枚举区间\([0,y),[y,2y),[2y,3y)...\)中的最小值(求后缀最小值也一样)更新答案,复 ...
-
BZOJ3351: [ioi2009]Regions(根号分治)
题意 题目链接 Sol 很神仙的题 我们考虑询问(a, b)(a是b的祖先),直接对b根号分治 如果b的出现次数\(< \sqrt{n}\),我们可以直接对每个b记录下与它有关的询问,这样每个询 ...
-
[CF587F]Duff is Mad[AC自动机+根号分治+分块]
题意 给你 \(n\) 个串 \(s_{1\cdots n}\) ,每次询问给出 \(l,r,k\) ,问在 \(s_{l\cdots r}\) 中出现了多少次 \(s_k\) . \(n,q,\su ...
-
[CF1039D]You Are Given a Tree[贪心+根号分治]
题意 给你\(n\)个点的树,其中一个简单路径的集合被称为\(k\)合法当且仅当树的每个节点最多属于一条路径,且每条路径包含\(k\)个节点.对于每个\(k(k \in [1,n])\),输出最多的\ ...
随机推荐
-
dhcp协议交互报文
DHCP共有八种报文,分别为DHCP Discover.DHCP Offer.DHCP Request.DHCP ACK.DHCP NAK.DHCP Release.DHCP Decline.DHCP ...
-
python学习笔记(一)
• Python能干嘛?[1]科学计算[2]图形化开发[3]系统脚本[4]Web服务器[5]网络爬虫[6]服务器集群自动化运维 • 常用工具:easy_install.pip.ipython.Subl ...
-
网站flash黑屏问题
操作系统 专业回答 2012-04-12 20:44 看网站视频时,可以小屏看,不能最大化.最大化的时候,只有声音,图象卡住了不动. 解决办法: 1 打开视频 然后最大化 按键 击右健 设置 把加速硬 ...
-
android 获取路径目录方法以及判断目录是否存在,创建目录
Environment 常用方法: * 方法:getDataDirectory()解释:返回 File ,获取 Android 数据目录.* 方法:getDownloadCacheDirectory( ...
-
OpenGL的学习与认识
OpenGL: Open Graphics Library 一套三维图形处理库,也是该领域的工业标准.是一个定义了一个跨编程语言,跨平台的编程接口规格的专业的图形程序接口.它用于三维图像(二维的亦可) ...
-
SELECT样式,兼容IE6
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
-
Unix System Overview
一.Unix 体系结构 由上图可以看出,内核居于最里层,Shell,Libary routines,以及Application通过系统调用(system calls)访问内核提供的功能.注意系统调用与 ...
-
配置Delphi工具菜单 转
配置Delphi工具菜单 Delphi工具菜单是可配置的.缺省时,Delphi Tools工具菜单的菜单项为[Database Desktop].[Image Editor].[Package Col ...
-
ASP.NET Core 使用 Alipay.AopSdk.Core 常见问题解答
1.Alipay.AopSdk.Core.AopException:"您使用的私钥格式错误,请检查RSA私钥配置,charset = UTF-8" 出现这个问题,就是配置不正确.首 ...
-
Go笔记-指针
Go 语言的取地址符是 &,放到一个变量前使用就会返回相应变量的内存地址 一个指针变量可以指向任何一个值的内存地址 它指向那个值的内存地址,在 32 位机器上占用 4 个字节,在 64 位机器 ...