题意:给你一个序列,问你这个序列有多少个子区间,满足把区间里的数排序之后相邻两个数之间的差 <= 1 ?

思路:https://blog.csdn.net/u013534123/article/details/81164504

代码:

#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
#define ls (o << 1)
#define rs (o << 1 | 1)
#define pii pair<int, int>
#define LL long long
using namespace std;
const int maxn = 100010;
map<int, int> last;
struct node {
int mi, add, cnt;
};
node tr[maxn * 4];
void maintain(pii& t1, pii t2) {
if(t1.first > t2.first) {
t1 = t2;
} else {
t1.second += t2.second;
}
}
void pushup(int o) {
if(tr[ls].mi < tr[rs].mi) {
tr[o].mi = tr[ls].mi;
tr[o].cnt = tr[ls].cnt;
} else if(tr[ls].mi > tr[rs].mi) {
tr[o].mi = tr[rs].mi;
tr[o].cnt = tr[rs].cnt;
} else {
tr[o].mi = tr[ls].mi;
tr[o].cnt = tr[ls].cnt + tr[rs].cnt;
}
}
void pushdown(int o) {
if(tr[o].add) {
tr[ls].mi += tr[o].add;
tr[ls].add += tr[o].add;
tr[rs].add += tr[o].add;
tr[rs].mi += tr[o].add;
tr[o].add = 0;
}
}
void build(int o, int l, int r) {
tr[o].mi = tr[o].add = tr[o].cnt = 0;
if(l == r) {
tr[o].cnt = 1;
return;
}
int mid = (l + r) >> 1;
build(ls, l, mid);
build(rs, mid + 1, r);
pushup(o);
}
void update(int o, int l, int r, int ql, int qr, int val) {
if(l >= ql && r <= qr) {
tr[o].add += val;
tr[o].mi += val;
return;
}
pushdown(o);
int mid = (l + r) >> 1;
if(ql <= mid) update(ls, l, mid, ql, qr, val);
if(qr > mid) update(rs, mid + 1, r, ql, qr, val);
pushup(o);
}
pii query(int o, int l, int r, int ql, int qr) {
if(l >= ql && r <= qr) {
return make_pair(tr[o].mi, tr[o].cnt);
}
int mid = (l + r) >> 1;
pushdown(o);
pii ans = make_pair(INF, 0);
if(ql <= mid) maintain(ans, query(ls, l, mid, ql, qr));
if(qr > mid) maintain(ans, query(rs, mid + 1, r, ql, qr));
return ans;
}
stack<int> mi, mx;
int a[maxn];
int main() {
int T, n;
LL ans = 0;
int kase = 0;
scanf("%d", &T);
while(T--) {
scanf("%d", &n);
for (int i = 1; i <= n; i++) {
scanf("%d", &a[i]);
}
// memset(tr, 0, sizeof(tr));
build(1, 1, n);
last.clear();
ans = 0;
while(mi.size())mi.pop();
while(mx.size())mx.pop();
mx.push(0), mi.push(0);
for (int i = 1; i <= n; i++) {
// update(1, 1, n, i, i, a[i]);
int now = i - 1;
while(mx.size() > 1 && a[mx.top()] <= a[i]) {
int tmp = mx.top();
mx.pop();
update(1, 1, n, mx.top() + 1, now, a[i] - a[tmp]);
now = mx.top();
}
mx.push(i);
// update(1, 1, n, i, i, -a[i]);
now = i - 1;
while(mi.size() > 1 && a[mi.top()] >= a[i]) {
int tmp = mi.top();
mi.pop();
update(1, 1, n, mi.top() + 1, now, a[tmp] - a[i]);
now = mi.top();
}
mi.push(i);
update(1, 1, n, last[a[i]] + 1, i, -1);
last[a[i]] = i;
pii tmp = query(1, 1, n, 1, i);
if(tmp.first == -1) {
ans += tmp.second;
}
}
printf("Case #%d: %lld\n", ++kase, ans);
}
}

  

2018ICPC银川 L Continuous Intervals 单调栈 线段树的更多相关文章

  1. 2018宁夏邀请赛 L Continuous Intervals(单调栈+线段树)

    2018宁夏邀请赛 L Continuous Intervals(单调栈+线段树) 传送门:https://nanti.jisuanke.com/t/41296 题意: 给一个数列A 问在数列A中有多 ...

  2. 洛谷P4198 楼房重建 单调栈+线段树

    正解:单调栈+线段树 解题报告: 传送门! 首先考虑不修改的话就是个单调栈板子题昂,这个就是 然后这题的话,,,我怎么记得之前考试好像有次考到了类似的题目昂,,,?反正我总觉着这方法似曾相识的样子,, ...

  3. 2018宁夏邀请赛 Continuous Intervals(单调栈 线段树

    https://vjudge.net/problem/Gym-102222L 题意:给你n个数的序列,让判断有几个区间满足排完序后相邻两数差都不大于1. 题解:对于一个区间 [L,R],记最大值为 m ...

  4. 南昌邀请赛I.Max answer 单调栈+线段树

    题目链接:https://nanti.jisuanke.com/t/38228 Alice has a magic array. She suggests that the value of a in ...

  5. The Preliminary Contest for ICPC China Nanchang National Invitational I. Max answer (单调栈+线段树)

    题目链接:https://nanti.jisuanke.com/t/38228 题目大意:一个区间的值等于该区间的和乘以区间的最小值.给出一个含有n个数的序列(序列的值有正有负),找到该序列的区间最大 ...

  6. 2019南昌网络赛-I(单调栈+线段树)

    题目链接:https://nanti.jisuanke.com/t/38228 题意:定义一段区间的值为该区间的和×该区间的最小值,求给定数组的最大的区间值. 思路:比赛时还不会线段树,和队友在这题上 ...

  7. 网络赛 I题 Max answer 单调栈+线段树

    题目链接:https://nanti.jisuanke.com/t/38228 题意:在给出的序列里面找一个区间,使区间最小值乘以区间和得到的值最大,输出这个最大值. 思路:我们枚举每一个数字,假设是 ...

  8. [CF1083D]The Fair Nut’s getting crazy[单调栈+线段树]

    题意 给定一个长度为 \(n\) 的序列 \(\{a_i\}\).你需要从该序列中选出两个非空的子段,这两个子段满足 两个子段非包含关系. 两个子段存在交. 位于两个子段交中的元素在每个子段中只能出现 ...

  9. 【CF671E】Organizing a Race 单调栈+线段树

    [CF671E]Organizing a Race 题意:n个城市排成一排,每个城市内都有一个加油站,赛车每次经过第i个城市时都会获得$g_i$升油.相邻两个城市之间由道路连接,第i个城市和第i+1个 ...

随机推荐

  1. shell时间转换脚本

    字符串转换为时间戳: time2utc #!/bin/sh Time=$ date -d "${Time}" '+%s' 时间戳转日期字符串 utc2time #!/bin/sh ...

  2. java导入ldif文件

    网上导入ldif文件的方式都是基于命令,或者相应工具如LDAP Browser \Editor v2.8.2. 但用java去实现这样的功能好像网上很少,于是我参照相应的开源代码并整理了一下,亲自测试 ...

  3. 11 November

    Weakness 求数列区间 \(\{a_n\}\) 中满足 \(i < j < k, a_i > a_j > a_k\) 的 \((i, j, k)\) 对的数目. 设对 \ ...

  4. 手把手教你搞定个推iOS推送SDK集成

    以下是一位开发者在集成个推iOS推送SDK过程中的真实经历. 作者:Ezreallp 一次偶然的机会,公司的项目要用到推送,我自己本来就很懒,不愿意去弄整套APNS的流程,刚好之前跟朋友聊起过他们的产 ...

  5. 走进JavaWeb技术世界7:Tomcat和其他WEB容器的区别

    本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutorial 喜欢的话麻烦点下 ...

  6. Mybatis, 实现一对多

    我这里是拿商品做为例子 不多说直接上代码 Mapper.xml <?xml version="1.0" encoding="UTF-8"?> < ...

  7. PJSIP Socket 模型

    前些日子为解决项目中问题调试到PJSIP Socket收发数据部分 ,好记性不如烂笔头,记录下 PJSIP 使用的Socket 模型 ,以备后用. 不同平台下PJSIP采用不同的 Socket模型,W ...

  8. Jmeter响应中中文乱码怎么解决?

    在jmeter的bin目录下有一个jmeter.properties的文件,打开它,搜索sampleresult.default.encoding,把它的注释打开,也就是把最前面的#去掉,改成samp ...

  9. 从进度条和alert的出现顺序来了解浏览器 UI 渲染 & JS进程

    项目里有一个需求是在上传文件的时候需要显示进度条,那么理所当然的在上传完成后就需要提示用户上传完毕并且更新进度条. 之前的预期表现是,上传完毕后,先更新进度条到100%,再alert出提示,所以代码如 ...

  10. curl 使用 post 请求,传递 json 参数,下载文件

    curl -X POST http://ip:8888/nacos/v1/cs/file/download -H "Accept: application/octet-stream" ...