题意

给一个长度为 \(n\) 的序列, \(q\) 次询问,次给一个 \(k_i\) ,问最少将序列划分成多少次,满足每一段的极差不超过\(w−k_i\).

\(1 \leq n, q \leq 10^5, 1 \leq w \leq 10^9,1 \leq k_i \leq w,0 \leq x_i \leq 10^9\)

分析

  • 每次直接贪心是正确的,可以考虑从第一段的影响证明,一定是尽量减少第二段的负担。

  • 记 \(k=w-k\) ,把询问按照 \(k\) 排序,那么段数显然单调不升。

  • 记 \({nxt}_i\) 表示 \(i\) 位置在当前询问的 \(k\) 下合法的最远位置 \(+1\) ,连边 \(i \rightarrow {nxt}_i\)。

  • 考虑在 \(k\) 变大的时候,我们修改一些位置的 \(nxt\),并使用 \(lct\) 加删边。如果 \({nxt}_i-i \geq \sqrt n\) 则不再连边。

    查询时如果走到了一棵树的树根便进行二分找到树根的 \(nxt\) ,因为二分时一定至少跳了 \(\sqrt n\) 步,所以这样的操作不会超过 \(\sqrt n\) 个。

  • 总时间复杂度为 \(O(n\sqrt nlogn)\)。

  • 感觉这种按数据根号分类的题都是平衡了两种暴力之间的复杂度

代码

#include<bits/stdc++.h>
using namespace std;
#define rep(i,a,b) for(int i=a;i<=b;++i)
#define pb push_back
typedef long long LL;
inline int gi(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-1;ch=getchar();}
while(isdigit(ch)){x=(x<<3)+(x<<1)+ch-48;ch=getchar();}
return x*f;
}
const int N=1e5 + 7;
int n,w,Q,sz;
int x[N],nxt[N],mi[N][20],mx[N][20],Log[N],ans[N];
struct qs{
int k,id;
bool operator <(const qs &rhs)const{
return k<rhs.k;
}
}q[N];
int fa[N],tr[N][2],son[N],rev[N];
#define pa fa[o]
#define Ls tr[o][0]
#define Rs tr[o][1]
bool isrt(int o){return tr[pa][0]^o&&tr[pa][1]^o;}
int side(int o){return tr[pa][1]==o;}
void pushup(int o){ son[o]=son[Ls]+son[Rs]+1; }
void rotate(int o){
int f=pa,y=fa[pa],x=side(o),s=tr[o][x^1];
if(!isrt(f)) tr[y][side(f)]=o;fa[o]=y;
tr[f][x]=s,fa[s]=f;
tr[o][x^1]=f,fa[f]=o;
pushup(f),pushup(o);
}
void splay(int o){
for(;!isrt(o);rotate(o))
if(!isrt(pa)) rotate(side(o)==side(pa)?pa:o);
}
void access(int o){
for(int y=0;o;y=o,o=pa) splay(o),Rs=y,pushup(o);
}
void link(int a,int b){
access(a),splay(a);
fa[a]=b;
}
void cut(int a,int b){
access(a),splay(a);
int x=tr[a][0];
fa[x]=tr[a][0]=0;
pushup(a);
}
int dep(int o){
access(o);splay(o);
return son[o];
}
int findr(int o){
access(o);splay(o);
while(Ls) o=Ls;
return o;
}
vector<int>G[N];
int qmx(int l,int r){
int k=Log[r-l+1];
return max(mx[l][k],mx[r-(1<<k)+1][k]);
}
int qmi(int l,int r){
int k=Log[r-l+1];
return min(mi[l][k],mi[r-(1<<k)+1][k]);
}
int main(){
n=gi(),w=gi(),Q=gi();sz=sqrt(n); rep(i,1,n) x[i]=mi[i][0]=mx[i][0]=gi();
x[n+1]=mi[n+1][0]=mx[n+1][0]=2e9+1; Log[1]=0; rep(i,2,n+1) Log[i]=Log[i>>1]+1;
for(int k=1;1<<k<=n+1;++k)
for(int i=1;i+(1<<k)-1<=n+1;++i){
mi[i][k]=min(mi[i][k-1],mi[i+(1<<k-1)][k-1]);
mx[i][k]=max(mx[i][k-1],mx[i+(1<<k-1)][k-1]);
} rep(i,1,Q){
q[i].id=i,q[i].k=w-gi();
} sort(q+1,q+1+Q);
rep(i,1,n+1) son[i]=1;
rep(i,1,n) nxt[i]=i,G[1].pb(i); rep(i,1,Q){
for(auto p:G[i]){
if(i!=1) cut(p,nxt[p]);
int j=nxt[p]+1;
for(;j<=min(p+sz,n+1);++j) if(qmx(p,j)-qmi(p,j)>q[i].k) break;
if(j==p+sz+1) continue; nxt[p]=j,link(p,j);
int x=lower_bound(q+1,q+1+Q,(qs){qmx(p,nxt[p])-qmi(p,nxt[p]),0})-q;
if(x!=Q+1) G[x].pb(p);
} int &res=ans[q[i].id];
for(int j=1;j<=n+1;){
res+=dep(j),j=findr(j);
if(j==n+1) break; int l=j+1,r=n+1;
while(l<r){
int mid=l+r>>1;
if(qmx(j,mid)-qmi(j,mid)>q[i].k) r=mid;
else l=mid+1;
}
j=l;
}
}
rep(i,1,Q) printf("%d\n",ans[i]-2);
return 0;
}

[CF1039E]Summer Oenothera Exhibition[根号分治+lct]的更多相关文章

  1. CF1039E Summer Oenothera Exhibition 根号分治,LCT,ST表

    CF1039E Summer Oenothera Exhibition LG传送门 根号分治好题. 可以先看我的根号分治总结. 题意就是给出长度为\(n\)的区间和\(q\)组询问以及一个\(w\), ...

  2. CF1039E Summer Oenothera Exhibition 贪心、根号分治、倍增、ST表

    传送门 感谢这一篇博客的指导(Orzwxh) $PS$:默认数组下标为$1$到$N$ 首先很明显的贪心:每一次都选择尽可能长的区间 不妨设$d_i$表示在取当前$K$的情况下,左端点为$i$的所有满足 ...

  3. Codeforces 1039D You Are Given a Tree [根号分治,整体二分,贪心]

    洛谷 Codeforces 根号分治真是妙啊. 思路 考虑对于单独的一个\(k\)如何计算答案. 与"赛道修建"非常相似,但那题要求边,这题要求点,所以更加简单. 在每一个点贪心地 ...

  4. BZOJ.4320.[ShangHai2006]Homework(根号分治 分块)

    BZOJ \(\mathbb{mod}\)一个数\(y\)的最小值,可以考虑枚举剩余系,也就是枚举区间\([0,y),[y,2y),[2y,3y)...\)中的最小值(求后缀最小值也一样)更新答案,复 ...

  5. BZOJ3351: [ioi2009]Regions(根号分治)

    题意 题目链接 Sol 很神仙的题 我们考虑询问(a, b)(a是b的祖先),直接对b根号分治 如果b的出现次数\(< \sqrt{n}\),我们可以直接对每个b记录下与它有关的询问,这样每个询 ...

  6. [CF587F]Duff is Mad[AC自动机+根号分治+分块]

    题意 给你 \(n\) 个串 \(s_{1\cdots n}\) ,每次询问给出 \(l,r,k\) ,问在 \(s_{l\cdots r}\) 中出现了多少次 \(s_k\) . \(n,q,\su ...

  7. [CF1039D]You Are Given a Tree[贪心+根号分治]

    题意 给你\(n\)个点的树,其中一个简单路径的集合被称为\(k\)合法当且仅当树的每个节点最多属于一条路径,且每条路径包含\(k\)个节点.对于每个\(k(k \in [1,n])\),输出最多的\ ...

  8. CF1039D You Are Given a Tree 根号分治,贪心

    CF1039D You Are Given a Tree LG传送门 根号分治好题. 这题可以整体二分,但我太菜了,不会. 根号分治怎么考虑呢?先想想\(n^2\)暴力吧.对于每一个要求的\(k\), ...

  9. 【bzoj5206】[Jsoi2017]原力 根号分治+STL-map

    题目描述 一个原力网络可以看成是一个可能存在重边但没有自环的无向图.每条边有一种属性和一个权值.属性可能是R.G.B三种当中的一种,代表这条边上原力的类型.权值是一个正整数,代表这条边上的原力强度.原 ...

随机推荐

  1. 用例设计之API用例覆盖准则

    基本原则 本文主要讨论API测试的用例/场景覆盖,基本原则如下: 用户场景闭环(从哪来到哪去) 遍历所有的实现逻辑路径 需求点覆盖 覆盖维度 API协议(参数&业务场景) 中间件检查 异常场景 ...

  2. Hibernate 注释用法

    注释 到现在为止,你已经看到 Hibernate 如何使用 XML 映射文件来完成从 POJO 到数据库表的数据转换的,反之亦然.Hibernate 注释是无需使用 XML 文件来定义映射的最新方法. ...

  3. c# 二分查找法

    1.仅 当 列表 是 有序 的 时候, 二分 查找 才 管用. 2.一般而言, 对于 包含 n 个 元素 的 列表, 用 二分 查找 最多 需要 log2n 步, 而 简单 查找 最多 需要 n 步. ...

  4. 如何在 Azure 门户中将托管数据磁盘附加到 Windows VM

    本文介绍了如何通过 Azure 门户将新的托管数据磁盘附加到 Windows 虚拟机. 在开始之前,请查看以下提示: 虚拟机的大小决定了可以附加多少个磁盘. 有关详细信息,请参阅虚拟机大小. 对于新磁 ...

  5. 可选的binlog解析组件

    本文的mysql-binlog-connector-java:https://github.com/shyiko/mysql-binlog-connector-java 阿里的canal:https: ...

  6. 转:.Net内存泄露原因及解决办法

    1.    什么是.Net内存泄露 (1).NET 应用程序中的内存 您大概已经知道,.NET 应用程序中要使用多种类型的内存,包括:堆栈.非托管堆和托管堆.这里我们需要简单回顾一下. 以运行库为目标 ...

  7. wc 命令使用说明

    wc 命令 使用说明 wc 命令还是很是简单的,通过 man 命令,可以见到可以选择的选项: wc option file 并且 wc 命令支持 管道操作 其中较为常用的命令选项 -c 字符的个数 - ...

  8. ln -s 软连接介绍

    软连接(softlink)也称符号链接.linux里的软连接文件就类似于windows系统中的快捷方式.软连接文件实际上是一个特殊的文件,文件类型是I.软连接文件实际上可以理解为一个文本文件,这个文件 ...

  9. xml 注意事项

      <?xml version="1.0" encoding="GB2312"?> xml区分大小写,只能有一个根元素,属性值必须放在引号中,空格不 ...

  10. js常见执行方法window.onload = function (){},$(document).ready()

    1. window.onload = function(){}; 当页面DOM对象加载完毕,web浏览器能够运行JS时,此方法即被触发. 2. $(document).ready();当web页面以及 ...