【题解】HNOI2016序列
也想了有半天,没有做出来……实际上做法确实也是十分精妙的。这里推荐一个blog,个人认为这位博主讲得挺好了:Sengxian's Blog;
感觉启示是:首先要加强对莫队算法 & ST表的熟练程度。 在想与数列有关的问题的时候,要尽量多在草稿纸上手玩几组数据,观察其中的联系,应该可以观察到许多有用的性质。
不过这题还有一个地方:网上的题解基本上都是先移动了右指针,后移动左指针;如果反过来,竟然被卡到只有10分。具体的原因我也不知道,但我猜想是不是由于r 和 l 的左右颠倒导致程序出现了一些问题,所以特判了一下,保证左指针永远在右指针的左侧。
#include <bits/stdc++.h>
using namespace std;
#define maxn 200000
#define int long long
int n, q, B = 316, a[maxn];
int ans, Ans[maxn];
int top, S[maxn], Log[maxn];
int LS[maxn], RS[maxn]; struct ques
{
int l, r, id, bel;
ques(int L = 0, int R = 0, int idx = 0)
{ l = L, r = R, id = idx, bel = (l / B) + 1; }
bool operator < (const ques &q)
const{
return (bel < q.bel || (bel == q.bel && r < q.r));
}
}Q[maxn]; int read()
{
int x = 0, k = 1;
char c;
c = getchar();
while(c < '0' || c > '9') { if(c == '-') k = -1; c = getchar(); }
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x * k;
} namespace RMQ
{
int ST[maxn][20];
int cmp(const int &i, const int &j) { return a[i] < a[j] ? i : j; }
void Work()
{
for(int i = 2; i <= n; i ++) Log[i] = Log[i >> 1] + 1;
for(int i = 1; i <= n; i ++) ST[i][0] = i;
for(int j = 1, k = 1 << j; k <= n; j ++, k = 1 << j)
for(int i = 1; k + i - 1 <= n; i ++)
ST[i][j] = cmp(ST[i][j - 1], ST[i + (k >> 1)][j - 1]);
}
int Get_min(int l, int r)
{
int k = Log[r - l + 1];
return cmp(ST[l][k], ST[r - (1 << k) + 1][k]);
}
} void Get_Sum(int *LS)
{
S[top = 0] = 0;
for(int i = 1; i <= n; i ++)
{
while(top && a[S[top]] >= a[i]) top --;
LS[i] = LS[S[top]] + (i - S[top]) * a[i];
S[++ top] = i;
}
} int Go_Right(int l, int r)
{
int pos = RMQ :: Get_min(l, r);
return (pos - l + 1) * a[pos] + LS[r] - LS[pos];
} int Go_Left(int l, int r)
{
int pos = RMQ :: Get_min(l, r);
return (r - pos + 1) * a[pos] + RS[l] - RS[pos];
} signed main()
{
n = read(), q = read(), B = sqrt(n) + 1;
for(int i = 1; i <= n; i ++) a[i] = read();
RMQ :: Work();
for(int i = 1; i <= q; i ++)
{
int x = read(), y = read();
Q[i] = ques(x, y, i);
}
sort(Q + 1, Q + 1 + q);
a[0] = -2e9; Get_Sum(LS); reverse(a + 1, a + 1 + n);
Get_Sum(RS); reverse(a + 1, a + 1 + n);
reverse(RS + 1, RS + 1 + n);
int l = 1, r = 1; ans = a[1];
for(int i = 1; i <= q; i ++)
{
if(Q[i].l > r)
{
while(r < Q[i].r) ans += Go_Right(l, r + 1), r ++;
while(r > Q[i].r) ans -= Go_Right(l, r), r --;
while(l < Q[i].l) ans -= Go_Left(l, r), l ++;
while(l > Q[i].l) ans += Go_Left(l - 1, r), l --;
}
else
{
while(l < Q[i].l) ans -= Go_Left(l, r), l ++;
while(l > Q[i].l) ans += Go_Left(l - 1, r), l --;
while(r < Q[i].r) ans += Go_Right(l, r + 1), r ++;
while(r > Q[i].r) ans -= Go_Right(l, r), r --;
}
Ans[Q[i].id] = ans;
}
for(int i = 1; i <= q; i ++) printf("%lld\n", Ans[i]);
return 0;
}
【题解】HNOI2016序列的更多相关文章
- 题解-[HNOI2016]序列
题解-[HNOI2016]序列 [HNOI2016]序列 给定 \(n\) 和 \(m\) 以及序列 \(a\{n\}\).有 \(m\) 次询问,每次给定区间 \([l,r]\in[1,n]\),求 ...
- 【LG3246】[HNOI2016]序列
[LG3246][HNOI2016]序列 题面 洛谷 题解 60pts 对于每个位置\(i\),单调栈维护它往左第一个小于等于它的位置\(lp_i\)以及往右第一个小于它的位置\(rp_i\). 那么 ...
- [BZOJ4540][HNOI2016]序列 莫队
4540: [Hnoi2016]序列 Time Limit: 20 Sec Memory Limit: 512 MB Description 给定长度为n的序列:a1,a2,…,an,记为a[1:n ...
- 【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]序列 题意:询问区间所有子串的最小值的和 不强制在线当然上莫队啦 但是没想出来,因为不知道该维护当前区间的什么信息,维护前后缀最小值的话不好做 想到单调栈求一下,但是对 ...
- 4540: [Hnoi2016]序列
4540: [Hnoi2016]序列 https://www.lydsy.com/JudgeOnline/problem.php?id=4540 分析: 莫队+RMQ+单调栈. 考虑加入一个点后,区间 ...
- BZOJ4540 Hnoi2016 序列 【莫队+RMQ+单调栈预处理】*
BZOJ4540 Hnoi2016 序列 Description 给定长度为n的序列:a1,a2,-,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,-,ar- ...
- [Bzoj4540][Hnoi2016] 序列(莫队 + ST表 + 单调队列)
4540: [Hnoi2016]序列 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1567 Solved: 718[Submit][Status] ...
- [HNOI2016]序列 CDQ+DP
[HNOI2016]序列 CDQ 链接 loj 思路 一个点最小变为l,最大变为r,不变的时候为v 那么j能在i前面就要满足. \(j<i\) \(r[j]<=v[i]\) \(v[j]& ...
- P6604 [HNOI2016]序列 加强版
*I. P6604 [HNOI2016]序列 加强版 摘自学习笔记 简单树论 笛卡尔树部分例题 I. 和 P6503 比较类似.我们设 \(f_i\) 表示全局以 \(i\) 结尾的子区间的最小值之和 ...
随机推荐
- 一步一步学习JNI
本文来自网易云社区 作者:孙有军 前言 本篇的主要目的就是JNI开发入门,使大家对JNI开发流程有一个大致的了解,后续再进行深入学习. JNI不是Android特有的,JNI是Java Native ...
- Codeforces-A. Shortest path of the king(简单bfs记录路径)
A. Shortest path of the king time limit per test 1 second memory limit per test 64 megabytes input s ...
- XX出行项目子系统-统计系统设计(定时器项目设计例子)
一. 引言 目前开发的XX出行系统,需要开发数据统计功能,鉴于约约出行系统已经在运营,并且有新版本的迭代,方便以后下个版本复用,遂新建一个子系统. 二. 架构设计 三. 具体实现 1.MySql数据库 ...
- linux c语言 fork() 和 exec 函数的简介和用法
linux c语言 fork() 和 exec 函数的简介和用法 假如我们在编写1个c程序时想调用1个shell脚本或者执行1段 bash shell命令, 应该如何实现呢? 其实在<std ...
- Docker学习笔记总结
Docker学习笔记 https://yeasy.gitbooks.io/docker_practice/content/ 一 环境搭建 Ubuntu安装 .添加软件源的GPG密钥 curl -f ...
- Greedy Gift Givers 贪婪的送礼者
Description 对于一群要互送礼物的朋友,TRW要确定每个人送出的钱比收到的多多少.在这一个问题中,每个人都准备了一些钱来送礼物,而这些钱将会被平均分给那些将收到他的礼物的人.然而,在任何一群 ...
- Java接口与继承作业
为什么子类的构造方法在运行之前,必须调用父类的构造方法?能不能反过来?为什么不能反过来? 因为子类继承了父类,那么就默认的含有父类的公共成员方法和公共成员变量,这些方法和变量在子类里不再重复声明.如果 ...
- ubuntu 安装 hustoj
https://github.com/zhblue/hustoj 准备工作: http://www.java123.net/v/961634.html 1.首先打开命令行,切换到root身份,获得最新 ...
- .net Mvc4 View—布局页与分部页
▲ 一.Layout属性 1.1.@RenderPage():渲染制定的页面到占位符. 注意:@RenderPage()可以使用多次,这点与@RenderBody()不同 ...
- Web界面和Winform界面生成,代码生成工具
在上面一篇随笔<代码生成工具之界面快速生成>介绍了代码生成工具Database2Sharp的界面生成操作,其中介绍了Web界面(包括列表界面.内容显示.内容编辑界面的生成,另外还介绍了Wi ...