4540: [Hnoi2016]序列


Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 1567  Solved: 718
[Submit][Status][Discuss]

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



Sample Output



HINT


1 ≤N,Q ≤ 100000,|Ai| ≤ 10^9

分析:


也许是我比较傻吧, 我把分块那里block[i] = i / tot +1打成了 block[i] = block[i] / tot + 1,超时了半天,调不出来……

题目又是没有修改,只有查询,可以联想到莫队算法。

题目要求是求一个大区间内,每个子区间最小值之和。我们可以看一下,如果已知一个区间[l,r],那么如果要得到[l,r + 1],我们会把[r + 1]的数添加进来。当这个数添加进来会多产生r - l + 2个子区间,那么如何处理这个子区间呢,就算是用st表 暴力去查询 这 r - l + 2个区间复杂度结合每一次也是很高的。

我们可以分析一下:如果我们找到[l,r + 1]里面的最小值,它所在位置为k。那么[l,k]这段区间的点为左端点和[r +1]所产生的 k - l + 1个区间都是以[k]所在数为最小值,那么贡献为(k - l + 1) * a[k],剩下的就是[k + 1,r + 1]这段区间了。

当我们知道一个数i左边第一个小于等于它的数的下标lm[i],所以[lm[i] + 1,i - 1]中间的值都是大于[i]的数的。所以以[lm[i] +1,i]的数都以[i]的数为最小值。设sumr[i]表示以i为右端点的合法区间的答案,那么sumr[i] = sumr[lm[i]] + (i  - lm[i]) * a[i]。发现我们的sumr数组是类似于前缀和的形式,最后我们用sum[r +1] - sum[k]就为[k + 1,r + 1]这段区间答案。

最终一个区间答案为ans =  (k - l + 1) * a[k] + sumr[r +1] - sum[k]。

删除同理,在左端点处添加删除只需维护rm[i]和suml[i]。求区间最小值的位置可以用st表,o(nlogn)的转移,o(1)的查询。求一个数左右第一个不大于它的数可以用单调队列,当一个数被另一个数踢出时,这个数离它最近的相应方向上的第一个不大于它的数就为另一个数。再加上莫队算法(nsqrt(n))总时间复制度为o(nlogn + nsqrt(n))


ST表:


类似于动态转移的思想,我们知道区间[l,r]的最小值,就可以推出区间[l,r + 1]的最小值,但是st表运用了倍增的思想(类似lca)。

我们知道了每个数向右2^(j - 1) 的最小值,就能拓展到每个数向右2 ^ j 的最小值。

转移方程为

st[i][j] = min(st[i][j - 1],st[i + (1 << j - 1)][j - 1]);

具体详见代码

附上AC代码:


# include <iostream>
# include <cmath>
# include <cstdio>
# include <cstring>
# include <algorithm>
using namespace std;
const int N = 2e5 + ;
int block[N],tot,n,m;
struct data{
int l,r,id;
bool operator <(const data & other)const{
if(block[l] == block[other.l])return r < other.r;
return block[l] < block[other.l];
}
}q[N];
int st[N][],a[N],mn[N];
int stack[N],cnt;
int lm[N],rm[N];
long long suml[N],sumr[N],num,ans[N];
int query(int x,int y){
return a[x] < a[y] ? x : y;
}
int Rmq(int L,int R)
{ if(L > R)swap(L,R);
int k = mn[R - L + ];
return query(st[L][k],st[R - ( << k) + ][k]);
}
void updatal(int x,int y,int z){
int pos = Rmq(x,y);
num +=(1ll * (y - pos + ) * a[pos] + sumr[x] - sumr[pos]) * z;
};
void updatar(int x,int y,int z){
int pos = Rmq(x,y);
num +=(1LL * (pos - x + ) * a[pos] + suml[y] - suml[pos]) * z;
};
void Modui(){
int L = ,R = ; num = ;
for (int i = ;i <= m;++i){
while (R < q[i].r) updatar(L,R + ,),R++;
while (L > q[i].l) updatal(L - ,R,),L--;
while (R > q[i].r) updatar(L,R,-),R--;
while (L < q[i].l) updatal(L,R,-),L++;
ans[q[i].id] = num;
}
}
void work(){
cnt = ;
for(int i = ;i <= n;i++){
while(cnt && a[stack[cnt]] >= a[i])rm[stack[cnt--]] = i;
stack[++cnt] = i;
}
while(cnt)rm[stack[cnt--]] = n + ;
for(int i = n;i >= ;i--){
while(cnt && a[stack[cnt]] >= a[i])lm[stack[cnt--]] = i;
stack[++cnt] = i;
}
for(int i = ;i <= n;i++){
suml[i] = suml[lm[i]] + 1LL * (i - lm[i]) * a[i];
} for(int i = n;i >= ;i--){
sumr[i] = sumr[rm[i]] + 1LL * (rm[i] - i) * a[i];
}
}
void init(){
mn[] = -;
for(int i = ;i <= n;i++){
mn[i]=((i & (i-))==) ? mn[i-]+ : mn[i-];
st[i][] = i;
}
for (int j = ;j <= mn[n];j++){
for (int i = ;i + ( << j) - <= n;i++){
st[i][j] = query(st[i][j - ],st[i + ( << j - )][j - ]);
}
}
work();
}
int main(){
scanf("%d %d",&n,&m);tot = sqrt(n);
for(int i = ;i <= n;i++){
scanf("%d",&a[i]);block[i] = i / tot + ;
}
init();
for(int i = ;i <= m;i++){
scanf("%d %d",&q[i].l,&q[i].r);
q[i].id = i;
}
sort(q + ,q + m + );
Modui();
for(int i = ;i <= m;i++){
printf("%lld\n",ans[i]);
}
return ;
}

[Bzoj4540][Hnoi2016] 序列(莫队 + ST表 + 单调队列)的更多相关文章

  1. BZOj 4540: [Hnoi2016]序列 [莫队 st表 预处理]

    4540: [Hnoi2016]序列 题意:询问区间所有子串的最小值的和 不强制在线当然上莫队啦 但是没想出来,因为不知道该维护当前区间的什么信息,维护前后缀最小值的话不好做 想到单调栈求一下,但是对 ...

  2. BZOJ4540 [Hnoi2016]序列 【莫队 + ST表 + 单调栈】

    题目 给定长度为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[ ...

  3. [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 ...

  4. [BZOJ4540][HNOI2016]序列 莫队

    4540: [Hnoi2016]序列 Time Limit: 20 Sec  Memory Limit: 512 MB Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n ...

  5. [HNOI2016]序列(莫队,RMQ)

    [HNOI2016]序列(莫队,RMQ) 洛谷  bzoj 一眼看不出来怎么用数据结构维护 然后还没修改 所以考虑莫队 以$(l,r-1) -> (l,r)$为例 对答案的贡献是$\Sigma_ ...

  6. Codeforces Round #278 (Div. 1) B - Strip dp+st表+单调队列

    B - Strip 思路:简单dp,用st表+单调队列维护一下. #include<bits/stdc++.h> #define LL long long #define fi first ...

  7. 【BZOJ4540】[Hnoi2016]序列 莫队算法+单调栈

    [BZOJ4540][Hnoi2016]序列 Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,…,a ...

  8. BZOJ.4540.[HNOI2016]序列(莫队/前缀和/线段树 单调栈 RMQ)

    BZOJ 洛谷 ST表的一二维顺序一定要改过来. 改了就rank1了哈哈哈哈.自带小常数没办法. \(Description\) 给定长为\(n\)的序列\(A_i\).\(q\)次询问,每次给定\( ...

  9. bzoj 4540 [HNOI 2016] 序列 - 莫队算法 - Sparse-Table - 单调栈

    题目传送门 传送点I 传送点II 题目大意 给定一个长度为$n$的序列.询问区间$[l, r]$的所有不同的子序列的最小值的和. 这里的子序列是连续的.两个子序列不同当且仅当它们的左端点或右端点不同. ...

随机推荐

  1. python 字符与字节 json序列和反序列及支持的类型

    b = b"demo" s = "demo" # 字符串转字节 s = bytes(s, encoding = "utf8") s = st ...

  2. 汇编2.汇编版本的helloworld

    寻址方式 立即数寻址 寄存器寻址 存储器寻址 直接寻址 : mov ax, [ 01000h ]; 直接在[]内给出一个内存地址 寄存器间接寻址: mov ax ,[si]; 在[]以寄存器的值给出内 ...

  3. 通过 getResources 找不到jar包中的资源和目录的解决方法

    http://my.oschina.net/sub/blog/184074 今天碰到一个怪问题: 原本跑的好好的代码,打成 jar 包就不能运行了. 问题出在,代码中有一段自动扫描 classpath ...

  4. DirectX9:高级着色语言(HLSL)

    一.简介 高级着色语言(High)可以编写顶点着色器和像素着色器,取代固定功能流水线中的部分功能,在图形卡的GPU(Graphics Processing Unit,图形处理单元)中执行 注意:如果图 ...

  5. C++11程序设计要点总结-模板机制详解

    C++程序设计要点总结 在编程的过程中呢我们总会遇到一些各种各样的问题,就比如在写方法的时候,我们一个同样的方法要写好几种类型的呢,这让我们很伤脑筋,但是呢C++有一个强大的功能就是模板机制,这个模板 ...

  6. vector性能调优之resize与reserve

    vector的resize与reserve reserve()函数为当前vector预留至少共容纳size个元素的空间.(译注:实际空间可能大于size) resize() 函数( void resi ...

  7. Manjaro/Arch linux 安装输入法

    输入命令: sudo pacman -Syu fcitx fcitx-googlepinyin fcitx-im fcitx-configtool 编辑 ~/.xprofile:# xfcitxexp ...

  8. systemverilog:task

    1.task declaration 个人喜欢ANSI C格式的声明: task mytask(output int x,input logic y); ...... endtask 注意端口列表的方 ...

  9. github 获取 token

    登录github 地址:https://github.com 点击settings 在点击Developer settings 继续 继续 描述栏随意写  复选框是token的权限 都选上吧 点击红框 ...

  10. 分享一款非常好用的Fatkun图片批量下载工具

    Fatkun图片批量下载 相信大家一定遇到过有着大量精美图片的网页,譬如美女照片.各种壁纸.设计素材.甚至是1024套图等等,但常常几十上百张的图要一张张手工去点击下载实在能让人抓狂!小编的工作中也常 ...