【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- ...
随机推荐
- 六、Django之Template
一.Template由来 1.任何前端页面的改动都和后端有关: 2.前端HTML和后端python分开能让网站更加清晰: 3.前后端分离的趋势下,专业的事交给专业的人做. 二.Django中的temp ...
- Migrating to WebSphere 9
Migrating to WebSphere 9 Make a migration plan Requirements Migrate WebSphere profiles into the new ...
- tomcat 设定自定义图片路径
1.问题 平常图片路径都是在项目目录下存放,都是ip地址+端口号+项目名+图片路径,因为项目需要要把图片从tomcat中分离出来,并且设置可以通过自定义地址访问自定义图片路径. 2.解决 在 tomc ...
- Python接口测试实战3(上)- Python操作数据库
如有任何学习问题,可以添加作者微信:lockingfree 课程目录 Python接口测试实战1(上)- 接口测试理论 Python接口测试实战1(下)- 接口测试工具的使用 Python接口测试实战 ...
- 使用calendar日历插件实现动态展示会议信息
效果图如下,标红色为有会议安排,并跳转详细会议信息页面. html页面 <%@ page contentType="text/html;charset=UTF-8"%> ...
- 无法设置主体sa的凭据
设置允许SQL Server身份登录 1.先用Window方式登陆进去,选择数据库实例,右键选择属性——安全性:把服务器身份验证选项从“Window身份验证模式”改为“SQLServer和Window ...
- python3【基础】-list&tuple
一.list概述 list (列表)是python中最常用的数据类型之一,通过列表可以对数据实现最方便的存储,修改等操作.在python3中,list支持如下方法: Help on class lis ...
- 基于spec评论作品 - 探路者 贪吃蛇
基于spec评论作品,试用(并截图)所有其他小组的Alpha作品,与软件功能说明书对比,评论Alpha作品对软件功能说明书的实现. 首先通过命令行进入到游戏主页面中. 因为软件没有编译为exe程序,所 ...
- 《C》变量
变量的存储方式和生存周期
- C# string 常用方法
string.ToString().Contains() String str="abcd" str.ToString().Contains("a"); //t ...