4540: [Hnoi2016]序列
4540: [Hnoi2016]序列
https://www.lydsy.com/JudgeOnline/problem.php?id=4540
分析:
莫队+RMQ+单调栈。
考虑加入一个点后,区间发生了什么变化。[l,r]->[l,r+1],增加了r-l+1段区间。设[l,r+1]的最小值在p,那么左端点在l~p-1的区间,答案就是a[p]了,p右边同样还存在许多最小值,影响了一段的区间。
每个点影响的区间是这样的,p的贡献就是a[p]*(p-l+1),剩余的贡献,可以记录每个位置向左的前缀和,那么求出p+1~R+1的贡献就行了(L可能不是最小的点,不能直接算L~R+1,只能从p开始算,p剩余的部分,单独算了)。同样的向右的也是这样求出。用单调栈求出每个位置左边第一个比它大的L[i],然后sl[i] = sl[L[i]] + a[i] * (i - L[i])。快速求出最小的值得位置的过程可以用RMQ维护。
于是就可以开心的用莫队了。
代码:
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<iostream>
#include<cctype>
#include<set>
#include<vector>
#include<queue>
#include<map>
#define fi(s) freopen(s,"r",stdin);
#define fo(s) freopen(s,"w",stdout);
using namespace std;
typedef long long LL; inline int read() {
int x=,f=;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-;
for(;isdigit(ch);ch=getchar())x=x*+ch-'';return x*f;
} const int N = ; int a[N], sk[N], bel[N], L[N], R[N], f[N][], Log[N];
LL ans[N], sl[N], sr[N], Answer;
struct Que{
int l, r, id;
bool operator < (const Que &A) const {
return bel[l] == bel[A.l] ? r < A.r : bel[l] < bel[A.l];
}
}q[N]; inline int Min(int i,int j) {
return a[i] < a[j] ? i : j;
}
inline int Calc(int l,int r) {
if (r < l) swap(l, r);
int t = Log[r - l + ];
return Min(f[l][t], f[r - ( << t) + ][t]);
}
inline void updl(int l,int r,int o) {
int p = Calc(l, r);
LL t = 1ll * a[p] * (r - p + ) + sr[l] - sr[p];
Answer += o * t;
}
inline void updr(int l,int r,int o) {
int p = Calc(l, r);
LL t = 1ll * a[p] * (p - l + ) + sl[r] - sl[p];
Answer += o * t;
}
int main() { fi("1.txt");
int n = read(), Q = read(), B = sqrt(n); Log[] = -;
for (int i=; i<=n; ++i) {
a[i] = read();
bel[i] = (i - ) / B + ;
Log[i] = Log[i >> ] + ;
f[i][] = i;
}
for (int i=; i<=Q; ++i)
q[i].l = read(), q[i].r = read(), q[i].id = i; for (int j=; j<=Log[n]; ++j)
for (int i=; i+(<<j)-<=n; ++i)
f[i][j] = Min(f[i][j - ], f[i + ( << (j - ))][j - ]); int top = ;
for (int i=; i<=n; ++i) {
while (top && a[sk[top]] >= a[i]) R[sk[top]] = i, top --;
sk[++top] = i;
}
while (top) R[sk[top]] = n + , top --;
for (int i=n; i>=; --i) {
while (top && a[sk[top]] > a[i]) L[sk[top]] = i, top --;
sk[++top] = i;
}
while (top) L[sk[top]] = , top --; for (int i=; i<=n; ++i) sl[i] = sl[L[i]] + 1ll * (i - L[i]) * a[i];
for (int i=n; i>=; --i) sr[i] = sr[R[i]] + 1ll * (R[i] - i) * a[i]; sort(q + , q + Q + );
int l = , r = ;
for (int i=; i<=Q; ++i) {
while (l > q[i].l) l --, updl(l, r, ); // 先加在减!!!
while (r < q[i].r) r ++, updr(l, r, );
while (l < q[i].l) updl(l, r, -), l ++;
while (r > q[i].r) updr(l, r, -), r --;
ans[q[i].id] = Answer;
}
for (int i=; i<=Q; ++i)
printf("%lld\n", ans[i]);
return ;
}
4540: [Hnoi2016]序列的更多相关文章
- BZOj 4540: [Hnoi2016]序列 [莫队 st表 预处理]
4540: [Hnoi2016]序列 题意:询问区间所有子串的最小值的和 不强制在线当然上莫队啦 但是没想出来,因为不知道该维护当前区间的什么信息,维护前后缀最小值的话不好做 想到单调栈求一下,但是对 ...
- BZOJ.4540.[HNOI2016]序列(莫队/前缀和/线段树 单调栈 RMQ)
BZOJ 洛谷 ST表的一二维顺序一定要改过来. 改了就rank1了哈哈哈哈.自带小常数没办法. \(Description\) 给定长为\(n\)的序列\(A_i\).\(q\)次询问,每次给定\( ...
- BZOJ 4540 [Hnoi2016]序列 | 莫队 详细题解
传送门 BZOJ 4540 题解 --怎么说呢--本来想写线段树+矩阵乘法的-- --但是嘛--yali的机房太热了--困--写不出来-- 于是弃疗,写起了莫队.(但是我连莫队都想不出来!) 首先用单 ...
- bzoj 4540: [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,则称 ...
- 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]序列 (单调栈 + ST表 + 莫队算法)
题目链接 BZOJ4540 考虑莫队算法. 这题难在$[l, r]$到$[l, r+1]$的转移. 根据莫队算法的原理,这个时候答案应该加上 $cal(l, r+1) + cal(l+1, r+1) ...
- bzoj 4540: [Hnoi2016]序列【单调栈+线段树】
强烈安利:http://blog.csdn.net/qq_34637390/article/details/51313126 这篇讲标记讲的非常好,这个标记非常神奇-- 首先last表示扫描到last ...
- [BZOJ4540][HNOI2016]序列 莫队
4540: [Hnoi2016]序列 Time Limit: 20 Sec Memory Limit: 512 MB Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n ...
- [Bzoj4540][Hnoi2016] 序列(莫队 + ST表 + 单调队列)
4540: [Hnoi2016]序列 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1567 Solved: 718[Submit][Status] ...
随机推荐
- 1085. [SCOI2005]骑士精神【IDA※】
Description 在一个5×5的棋盘上有12个白色的骑士和12个黑色的骑士, 且有一个空位.在任何时候一个骑士都能按照骑 士的走法(它可以走到和它横坐标相差为1,纵坐标相差为2或者横坐标相差为2 ...
- 如何彻底修改eclipse中的名称
一.右键工程:Refactor->Rename,或选中工程按F2,修改名称二.修改项目目录下:.project文件 <?xml version="1.0" encodi ...
- python执行linux和window的命令
linux: python执行shell脚本常用的方法 import os val=os.system("shell语句") >>> val=os.system ...
- 【题解】洛谷P1966 [NOIP2013TG] 火柴排队(树状数组+逆序对)
次元传送门:洛谷P1966 思路 显然在两排中 每排第i小的分别对应就可取得最小值(对此不给予证明懒) 所以我们只在意两排的火柴是第几根 高度只需要用来进行排序(先把两个序列改成有序的方便离散化) 因 ...
- 软工之词频统计器及基于sketch在大数据下的词频统计设计
目录 摘要 算法关键 红黑树 稳定排序 代码框架 .h文件: .cpp文件 频率统计器的实现 接口设计与实现 接口设计 核心功能词频统计器流程 效果 单元测试 性能分析 性能分析图 问题发现 解决方案 ...
- centos6.4安装使用wine 持续更新中
首先,从wine的官网下载页面http://www.winehq.org/download/可以了解到centos安装wine需要EPEL软件仓库.那么首先安装EPEL软件仓库,从http://mir ...
- Knowledge Point 20180506 深究Java的跨平台特性
本章主题:从骨子里看Java的跨平台;本文内容部分摘自https://www.cnblogs.com/roger-yu/p/5827452.html 有过基础Java知识的开发人员都知道Java是跨平 ...
- Android 心跳呼吸动画
废话少说,看东西 一个很简单的心跳呼吸的动画,几行代码搞定: 代码: private ImageView ivHart; //图片 AlphaAnimation alphaAnimation = nu ...
- oracle数据库每天自动备份
现在介绍一下我们项目中使用exp工具每天自动对oracle进行备份. 用这个工具我们可以使用命令行的方式也可以使用pl/sql developer来进行导出. 我们要使用exp命令来导出oracle的 ...
- 25条提高iOS App性能的技巧和诀窍
25条提高iOS App性能的技巧和诀窍 当我们开发iOS应用时,好的性能对我们的App来说是很重要的.你的用户也希望如此,但是如果你的app表现的反应迟钝或者很慢也会伤害到你的审核. 然而,由于IO ...