[HNOI2016]序列(莫队,RMQ)
[HNOI2016]序列(莫队,RMQ)
一眼看不出来怎么用数据结构维护
然后还没修改
所以考虑莫队
以$(l,r-1) -> (l,r)$为例
对答案的贡献是$\Sigma_{i=l}^{r}minval(a[i:r])$
考虑维护前缀和
先用单调栈扫出$w[i]$作为最小值的左边界右边界$l_i,r_i$
然后回到上面的例子贡献就是$frontsum_{r}-frontsum_{rmqmin(l,r)}+w[rmqmin(l,r)]*(rmqmin(l,r)-l+1)$
完结...等等我好像忘了点啥
$frontsum_{i}$指的是从i向前拓展的i个包含i的子区间的区间最小值的和,也是可以用单调栈求出来的
时间复杂度略
#include<cstdio>
#include<cmath>
#include<algorithm>
using std::sort;
typedef long long lint;
const int N=100011;
template<typename tp>void read(tp &kk){
#define ak *
tp phy=0,ioi=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')ioi=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){phy=phy*10+ch-'0';ch=getchar();}
kk=phy ak ioi;
}
int n,m,bl[N];
lint w[N];
int bsize; int lg[N];
int lgl[N][17];
void is()
{
for(int i=2;i<=n;i++) lg[i]=lg[i>>1]+1;
for(int i=1;i<=n;i++) lgl[i][0]=i;
for(int k=1;k<=lg[n];k++)
{
for(int i=1;i-1+(1<<k)<=n;i++)
{
lgl[i][k]=w[lgl[i][k-1]]<w[lgl[i+(1<<(k-1))][k-1]]?lgl[i][k-1]:lgl[i+(1<<(k-1))][k-1];
}
}
}
int rmq(int l,int r){
return w[lgl[l][lg[r-l+1]]]<
w[lgl[r-(1<<lg[r-l+1])+1][lg[r-l+1]]]?
lgl[l][lg[r-l+1]]:
lgl[r-(1<<lg[r-l+1])+1][lg[r-l+1]];
} int li[N],ri[N];
int st[N],hop;
void rkk()
{
for(int i=1;i<=n;i++)
{
while(hop&&w[st[hop]]>w[i])
ri[st[hop]]=i-1,hop--;
st[++hop]=i;
if(i==n)
while(hop) ri[st[hop]]=i,hop--;
}
for(int i=n;i;i--)
{
while(hop&&w[st[hop]]>w[i])
li[st[hop]]=i+1,hop--;
st[++hop]=i;
if(i==1)
while(hop) li[st[hop]]=i,hop--;
}
}
lint fsum[N],bsum[N];
void baka()
{
for(int i=1;i<=n;i++)
fsum[i]=fsum[li[i]-1]+w[i]*(i-li[i]+1);
for(int i=n;i;i--)
bsum[i]=bsum[ri[i]+1]+w[i]*(ri[i]-i+1);
} struct ques
{
int l,r,id;
bool friend operator < (ques a,ques b)
{
if(bl[a.l]!=bl[b.l]) return bl[a.l]<bl[b.l];
return bl[a.r]<bl[b.r];
}
}q[N]; lint prt[N],tmp; int main()
{
read(n),read(m),bsize=ceil(sqrt(n));
for(int i=1;i<=n;i++) read(w[i]),bl[i]=(i-1)/bsize+1;
rkk();
is();
baka();
for(int i=1;i<=m;i++) read(q[i].l),read(q[i].r),q[i].id=i;
sort(q+1,q+1+m);
int l=1,r=1;tmp=w[1];
for(int i=1;i<=m;i++)
{
while(r<q[i].r){r++;tmp+=fsum[r]-fsum[rmq(l,r)]+((rmq(l,r)-l+1)*w[rmq(l,r)]);}
while(l>q[i].l){l--;tmp+=bsum[l]-bsum[rmq(l,r)]+((r-rmq(l,r)+1)*w[rmq(l,r)]);}
while(r>q[i].r){tmp-=fsum[r]-fsum[rmq(l,r)]+((rmq(l,r)-l+1)*w[rmq(l,r)]);r--;}
while(l<q[i].l){tmp-=bsum[l]-bsum[rmq(l,r)]+((r-rmq(l,r)+1)*w[rmq(l,r)]);l++;}
prt[q[i].id]=tmp;
}
for(int i=1;i<=m;i++) printf("%lld\n",prt[i]);
return 0;
}
[HNOI2016]序列(莫队,RMQ)的更多相关文章
- BZOJ.4540.[HNOI2016]序列(莫队/前缀和/线段树 单调栈 RMQ)
BZOJ 洛谷 ST表的一二维顺序一定要改过来. 改了就rank1了哈哈哈哈.自带小常数没办法. \(Description\) 给定长为\(n\)的序列\(A_i\).\(q\)次询问,每次给定\( ...
- BZOj 4540: [Hnoi2016]序列 [莫队 st表 预处理]
4540: [Hnoi2016]序列 题意:询问区间所有子串的最小值的和 不强制在线当然上莫队啦 但是没想出来,因为不知道该维护当前区间的什么信息,维护前后缀最小值的话不好做 想到单调栈求一下,但是对 ...
- 【BZOJ4540】[Hnoi2016]序列 莫队算法+单调栈
[BZOJ4540][Hnoi2016]序列 Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,…,a ...
- [BZOJ4540][HNOI2016]序列 莫队
4540: [Hnoi2016]序列 Time Limit: 20 Sec Memory Limit: 512 MB Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n ...
- [bzoj4540][Hnoi2016][序列] (莫队算法+单调栈+st表)
Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,…,ar-1,ar.若1≤l≤s≤t≤r≤n,则称a ...
- 洛谷P3246 [HNOI2016]序列 [莫队]
传送门 思路 看到可离线.无修改.区间询问,相信一定可以想到莫队. 然而,莫队怎么转移是个大问题. 考虑\([l,r]\rightarrow[l,r+1]\)时答案会怎样变化?(左端点变化时同理) \ ...
- bzoj 4540: [Hnoi2016]序列 莫队
题目: 给定长度为n的序列:a1,a2,-,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,-,ar- 1,ar.若1≤l≤s≤t≤r≤n,则称a[s:t]是a ...
- BZOJ 4540 [Hnoi2016]序列 | 莫队 详细题解
传送门 BZOJ 4540 题解 --怎么说呢--本来想写线段树+矩阵乘法的-- --但是嘛--yali的机房太热了--困--写不出来-- 于是弃疗,写起了莫队.(但是我连莫队都想不出来!) 首先用单 ...
- 【BZOJ4540】【HNOI2016】序列 [莫队][RMQ]
序列 Time Limit: 20 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description 给定长度为n的序列:a1,a2,…,a ...
随机推荐
- Solution -「洛谷 P4719」「模板」"动态 DP" & 动态树分治
\(\mathcal{Description}\) Link. 给定一棵 \(n\) 个结点的带权树,\(m\) 次单点点权修改,求出每次修改后的带权最大独立集. \(n,m\le10^5 ...
- php发送邮件 (phpmailer)
1.首先下载phpMailer文件官方文件https://sourceforge.net/projects/phpmailer/: 还有class.smtp.php. 2.去配置一下发送邮件的服务器, ...
- Redis 源码简洁剖析 13 - RDB 文件
RDB 是什么 RDB 文件格式 Header Body DB Selector AUX Fields Key-Value Footer 编码算法说明 Length 编码 String 编码 Scor ...
- Java邮件发送中的setRecipient方法使用
一.方法setRecipient(Message.RecipientType type, Address address),是用于设置邮件的接收者. 1.有两个参数,第一个参数是接收者的类型,第二 ...
- KVM 虚机镜像操作, 扩容和压缩
KVM镜像操作 qemu-img命令 创建镜像 qemu-img create # 创建一个设备空间大小为10G的镜像 qemu-img create -f qcow2 centos7-guest.q ...
- 攻防世界Web_shrine
题目: 给的是源代码,整理一下如下: 看到jinjia flask,render_template_string猜测到这题应该是考查python模板注入. 代码分析可以得到以下信息: 1.路径在 /s ...
- 技术管理进阶——为什么Leader的话有时候你听不懂
原创不易,求分享.求一键三连 Hi,各位亲爱的小伙伴,小钗公号遵循日复盘->周复盘->月复盘->季度复盘->年总结策略,所以某类型文章到后期才会成体系. 今天这篇文章属于「月复 ...
- Zabbix是什么?
概述 Zabbix 是一个企业级的分布式开源监控方案,可以监控服务器健康性以及网络参数的一款软件,Zabbix几乎可以为任何时间配置邮件警告,这样用户可以实时通过邮箱接收服务器所发生的任何问题.对于已 ...
- Python 的垃圾回收
垃圾回收 首先介绍两个画图的工具:objgraph 包和在线绘图网站 draw.io.具体的使用以后再写. 1.引用计数 Python 中,每个对象都有存有指向该对象的引用总数,即:引用计数(refe ...
- VSCode好用的插件
一.Chinese: 汉化 二.Live Server: Live Serve这是一个为静态和动态页面启动具有实时重新加载功能的开发本地服务器. 安装好后,在html页面右键:Open With Li ...