【BZOJ4540】【HNOI2016】序列(莫队)
【BZOJ4540】【HNOI2016】序列(莫队)
题面
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[s:t]是a[l:r]的子序列。现在有q个询问,每个询问给定两个数l和r,1≤l≤r
≤n,求a[l:r]的不同子序列的最小值之和。例如,给定序列5,2,4,1,3,询问给定的两个数为1和3,那么a[1:3]有
6个子序列a[1:1],a[2:2],a[3:3],a[1:2],a[2:3],a[1:3],这6个子序列的最小值之和为5+2+4+2+2+2=17。
Input
输入文件的第一行包含两个整数n和q,分别代表序列长度和询问数。接下来一行,包含n个整数,以空格隔开
,第i个整数为ai,即序列第i个元素的值。接下来q行,每行包含两个整数l和r,代表一次询问。
Output
对于每次询问,输出一行,代表询问的答案。
Sample Input
5 5
5 2 4 1 3
1 5
1 3
2 4
3 5
2 5
Sample Output
28
17
11
11
17
HINT
1 ≤N,Q ≤ 100000,|Ai| ≤ 10^9
题解
我其实本来不想写莫队来着
但是在网上找题解都是莫队
无奈。。。我也写莫队。。
莫队的重点就在于怎么\(O(1)\)转移状态
假设我们已经求出了\([L+1,R]\)的答案
现在要扩展到\([L,R]\)
考虑新产生的\([L..L],[L..L+1]...,[L...R]\)的答案
我们先找到这段区间的最小值,假设其位置是\(p\)
那么右端点在\([p,R]\)的子序列的贡献都是\(a[p]\)
接下来呢?把\([L,p-1]\)继续考虑?
但是我们要做到转移\(O(1)\),所以考虑怎么优化
我们设\(f[i]\)表示确定左端点为\(i\)时,到后面所有位置的贡献
利用单调栈求出右侧第一个比\(i\)位置小的数的位置\(R[i]\)
\([i,R[i]-1]\)的贡献就是\(a[i]\),而\([R[i],n]\)的贡献则与\(i\)无关,
只与\(R[i]\)有关,因此,我们得到转移
\]
所以,此时\([L,p)\)的贡献就是\(f[L]-f[p]\)
考虑右侧的贡献同理
综上,时间复杂度\(O(n\sqrt{n})\)
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 111111
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
int n,q,blk;
int L[MAX],R[MAX],S[MAX],top,a[MAX],lg[MAX];
ll f[MAX],g[MAX],ans[MAX],Ans;
struct Query{int i,l,r,blk;}Q[MAX];
bool operator<(Query a,Query b){if(a.blk!=b.blk)return a.blk<b.blk;return a.r<b.r;}
struct STable
{
int p[18][MAX];
void pre()
{
for(int j=1;j<=lg[n];++j)
for(int i=1;i+(1<<(j-1))<=n;++i)
p[j][i]=a[p[j-1][i]]<=a[p[j-1][i+(1<<(j-1))]]?p[j-1][i]:p[j-1][i+(1<<(j-1))];
}
int Query(int l,int r)
{
int k=lg[r-l+1];
return a[p[k][l]]<=a[p[k][r-(1<<k)+1]]?p[k][l]:p[k][r-(1<<k)+1];
}
}ST;
ll CalcL(int l,int r)
{
int p=ST.Query(l,r);
return 1ll*(r-p+1)*a[p]+g[l]-g[p];
}
ll CalcR(int l,int r)
{
int p=ST.Query(l,r);
return 1ll*(p-l+1)*a[p]+f[r]-f[p];
}
int main()
{
n=read();q=read();blk=sqrt(n);
for(int i=1;i<=n;++i)a[i]=read();
for(int i=1;i<=n;++i)
{
while(top&&a[S[top]]>a[i])--top;
L[i]=S[top];
S[++top]=i;
}
for(int i=1;i<=n;++i)f[i]=f[L[i]]+1ll*(i-L[i])*a[i];
S[top=0]=n+1;
for(int i=n;i;--i)
{
while(top&&a[S[top]]>a[i])--top;
R[i]=S[top];
S[++top]=i;
}
for(int i=n;i>=1;--i)g[i]=g[R[i]]+1ll*(R[i]-i)*a[i];
for(int i=1;i<=q;++i)
{
int l=read(),r=read();
Q[i]=(Query){i,l,r,(l-1)/blk};
}
for(int i=1;i<=n;++i)ST.p[0][i]=i;
for(int i=2;i<=n;++i)lg[i]=lg[i>>1]+1;
ST.pre();
sort(&Q[1],&Q[q+1]);
for(int i=1,L=1,R=0;i<=q;++i)
{
while(R<Q[i].r)Ans+=CalcR(L,++R);
while(L>Q[i].l)Ans+=CalcL(--L,R);
while(R>Q[i].r)Ans-=CalcR(L,R--);
while(L<Q[i].l)Ans-=CalcL(L++,R);
ans[Q[i].i]=Ans;
}
for(int i=1;i<=q;++i)printf("%lld\n",ans[i]);
return 0;
}
【BZOJ4540】【HNOI2016】序列(莫队)的更多相关文章
- [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 ...
- [HNOI2016]序列(莫队,RMQ)
[HNOI2016]序列(莫队,RMQ) 洛谷 bzoj 一眼看不出来怎么用数据结构维护 然后还没修改 所以考虑莫队 以$(l,r-1) -> (l,r)$为例 对答案的贡献是$\Sigma_ ...
- 【BZOJ4540】[Hnoi2016]序列 莫队算法+单调栈
[BZOJ4540][Hnoi2016]序列 Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,…,a ...
- BZOj 4540: [Hnoi2016]序列 [莫队 st表 预处理]
4540: [Hnoi2016]序列 题意:询问区间所有子串的最小值的和 不强制在线当然上莫队啦 但是没想出来,因为不知道该维护当前区间的什么信息,维护前后缀最小值的话不好做 想到单调栈求一下,但是对 ...
- BZOJ.4540.[HNOI2016]序列(莫队/前缀和/线段树 单调栈 RMQ)
BZOJ 洛谷 ST表的一二维顺序一定要改过来. 改了就rank1了哈哈哈哈.自带小常数没办法. \(Description\) 给定长为\(n\)的序列\(A_i\).\(q\)次询问,每次给定\( ...
- 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 ...
- 洛谷P3246 [HNOI2016]序列 [莫队]
传送门 思路 看到可离线.无修改.区间询问,相信一定可以想到莫队. 然而,莫队怎么转移是个大问题. 考虑\([l,r]\rightarrow[l,r+1]\)时答案会怎样变化?(左端点变化时同理) \ ...
- BZOJ 4540 [Hnoi2016]序列 | 莫队 详细题解
传送门 BZOJ 4540 题解 --怎么说呢--本来想写线段树+矩阵乘法的-- --但是嘛--yali的机房太热了--困--写不出来-- 于是弃疗,写起了莫队.(但是我连莫队都想不出来!) 首先用单 ...
- BZOJ4540 Hnoi2016 序列 【莫队+RMQ+单调栈预处理】*
BZOJ4540 Hnoi2016 序列 Description 给定长度为n的序列:a1,a2,-,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,-,ar- ...
随机推荐
- elasticsearch对某段时间范围内按时间间隔进行统计
{ "query" : { "constant_score" : { "filter" : { "range" : { ...
- python全栈开发-前方高能-函数
python_day_9 一.今日主要内容 函数: 函数:对功能的封装 语法: def 函数名(形参): 函数体 函数名(实参) 函数名:命名规则和变量一样 函数的返回值: return, 函数执行完 ...
- shell loop
#!/bin/sh date i=0 while [ $i -le 30 ] do echi $i /usr/sbin/r2/np_test_acl -f rule.txt i=$(e ...
- TPO-17 C2 Reschedule part-time job in campus dining hall
TPO-17 C2 Reschedule part-time job in campus dining hall 第 1 段 1.Listen to a conversation between a ...
- hbase和ZooKeeper集群安装配置
一:ZooKeeper集群安装配置 1:解压zookeeper-3.3.2.tar.gz并重命名为zookeeper. 2:进入~/zookeeper/conf目录: 拷贝zoo_sample.cfg ...
- Maven私库
<server> <id>releases</id> <username>admin</username> <password> ...
- XSS 注入检查点
如果你有个论坛,一般你会很注意用户发帖的注入问题,往往这个地方不会被注入,因为开发特别照顾.原则上XSS都是用户输入的,但是许多边角还是容易忽略.枚举一些检查点. 分页 分页通用组件获取url,修改p ...
- Tess4J -4.0.2- Linux 实践 [解决:Tess4J - Native library (linux-x86-64/libtesseract.so) not found in resource path]
[本文编写于2018年7月5日] Tess4J是Tesseract的Java JNA wrapper.本文介绍了在CentOS 7 操作系统中使用Tess4J的步骤及注意事项.在正式开始之前,先花一点 ...
- 下载android sdk更新包离线安装解决方案
本文转载自:http://xljboox.blog.163.com/blog/static/7628448320111159354738/ 第一次安装android sdk后进行开发包的更新,你应该了 ...
- [linux] centos搭建openvpn服务,脚本颁发/吊销证书 (转载+原创)
搭建过程转载:http://yestreenstars.blog.51cto.com/1836303/1429537 环境说明:服务端:CentOS 6.5_X64客户端:Windows 7 服务端配 ...