2018ICPC银川 L Continuous Intervals 单调栈 线段树

题意:给你一个序列,问你这个序列有多少个子区间,满足把区间里的数排序之后相邻两个数之间的差 <= 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);
	}
} 

  

posted @ 2019-09-03 17:12  维和战艇机  阅读(269)  评论(0编辑  收藏  举报