JZOJ4605. 排序(线段树合并与分裂)
题目大意:
每次把一个区间升序或降序排序,最后问一个点是什么。
题解:
如果只是问一个点,这确乎是个经典题,二分一下答案然后线段树维护01排序。
从pty那里get到了可以用线段树的合并与分裂实时地维护整个序列。
考虑一次排序就把这个区间的数搞到一个线段树上,在根处标记是正的还是反的。
如果想搞到一棵树上就需要用到分裂与合并,根据势能分析,复杂度还是\(O(n~log~n)\)。
Code:
#include<bits/stdc++.h>
#define fo(i, x, y) for(int i = x, B = y; i <= B; i ++)
#define ff(i, x, y) for(int i = x, B = y; i < B; i ++)
#define fd(i, x, y) for(int i = x, B = y; i >= B; i --)
#define ll long long
#define pp printf
#define hh pp("\n")
using namespace std;
const int N = 1e5 + 5;
int n, m, a[N];
int op, l, r;
struct tree {
int l, r, x;
} t[N * 120];
#define i0 t[i].l
#define i1 t[i].r
int tot, pl, pr, px;
int rt[N * 120], rev[N * 120];
void upd(int i) {
t[i].x = t[i0].x + t[i1].x;
}
void add(int &i, int x, int y) {
i = ++ tot;
t[i].x = 1;
if(x == y) return;
int m = x + y >> 1;
if(pl <= m) add(i0, x, m); else add(i1, m + 1, y);
}
void merge(int &i, int j) {
if(!i || !j) { i = i + j; return;}
merge(t[i].l, t[j].l);
merge(t[i].r, t[j].r);
upd(i);
}
void ft(int i, int x, int y) {
if(x == y) { if(t[i].x == pl) px = x; return;}
int m = x + y >> 1;
if(t[i0].x >= pl) ft(i0, x, m); else
pl -= t[i0].x, ft(i1, m + 1, y);
}
void sp(int i, int &j1, int &j2, int x, int y, int k) {
if(!i) return;
if(y <= k) { j1 = i, j2 = 0; return;}
if(x > k) { j1 = 0, j2 = i; return;}
int m = x + y >> 1;
if(!j1) j1 = ++ tot;
if(!j2) j2 = ++ tot;
sp(i0, t[j1].l, t[j2].l, x, m, k);
sp(i1, t[j1].r, t[j2].r, m + 1, y, k);
upd(j1); upd(j2);
}
int bz[N];
void dg(int i, int x, int y) {
if(!t[i].x) return;
if(x == y) { bz[x] ++; return;}
int m = x + y >> 1;
dg(i0, x, m); dg(i1, m + 1, y);
}
void split(int &i, int &j, int k) {
pl = rev[i] ? t[i].x - k : k;
px = 0; ft(i, 0, n);
if(rev[i]) sp(i, j, i, 0, n, px); else sp(i, i, j, 0, n, px);
}
set<int> s;
set<int> :: iterator st, en;
int d[N], d0;
int main() {
scanf("%d %d", &n, &m);
fo(i, 1, n) scanf("%d", &a[i]);
fo(i, 0, n + 1) s.insert(i);
fo(i, 0, n) pl = pr = a[i], add(rt[i], 0, n);
fo(i, 1, m) {
scanf("%d %d %d", &op, &l, &r);
en = s.upper_bound(r);
st = -- s.lower_bound(l);
d0 = 0;
for(; st != en; st ++) d[++ d0] = *st;
d[++ d0] = *en;
int rt1 = 0, rt2 = 0;
split(rt[d[1]], rt1, l - d[1]);
if(t[rt1].x) {
rt[l] = rt1;
rev[rt[l]] = rev[rt[d[1]]];
d[1] = l;
}
if(d[d0] != r + 1) {
split(rt[d[d0 - 1]], rt2, r + 1 - d[d0 - 1]);
rt[r + 1] = rt2;
rev[rt[r + 1]] = rev[rt[d[d0 - 1]]];
}
fo(j, 2, d0 - 1) {
if(l != d[j]) merge(rt[l], rt[d[j]]), rt[d[j]] = 0;
s.erase(d[j]);
}
rev[rt[l]] = op;
s.insert(l); s.insert(r + 1);
}
scanf("%d", &l);
l ++;
for(st = s.begin(); st != s.end(); st ++) {
int i = *st;
if(t[rt[i]].x >= l) {
pl = rev[rt[i]] ? t[rt[i]].x - l + 1 : l;
px = 0;
ft(rt[i], 0, n);
pp("%d\n", px);
return 0;
} else l -= t[rt[i]].x;
}
}
JZOJ4605. 排序(线段树合并与分裂)的更多相关文章
- BZOJ.4552.[HEOI2016/TJOI2016]排序(线段树合并/二分 线段树)
题目链接 对于序列上每一段连续区间的数我们都可以动态开点建一棵值域线段树.初始时就是\(n\)棵. 对于每次操作,我们可以将\([l,r]\)的数分别从之前它所属的若干段区间中分离出来,合并. 对于升 ...
- BZOJ4552 HEOI2016/TJOI2016排序(线段树合并+线段树分裂)
很久以前写过二分答案离线的做法,比较好理解.事实上这还是一个线段树合并+分裂的板子题,相比离线做法以更优的复杂度做了更多的事情.具体不说了.怎么交了一遍luogu上就跑第一了啊 #include< ...
- 有趣的线段树模板合集(线段树,最短/长路,单调栈,线段树合并,线段树分裂,树上差分,Tarjan-LCA,势能线段树,李超线段树)
线段树分裂 以某个键值为中点将线段树分裂成左右两部分,应该类似Treap的分裂吧(我菜不会Treap).一般应用于区间排序. 方法很简单,就是把分裂之后的两棵树的重复的\(\log\)个节点新建出来, ...
- 【线段树合并】【P2824】 [HEOI2016/TJOI2016]排序
Description 给定一个长度为 \(n\) 的排列,有 \(m\) 次操作,每次选取一段局部进行升序或降序排序,问你一波操作后某个位置上的数字是几 Hint \(1~\leq~n,~m~\le ...
- 洛谷 P2824 [HEOI2016/TJOI2016]排序 (线段树合并)
(另外:题解中有一种思路很高妙而且看上去可以适用一些其他情况的离线方法) 线段树合并&复杂度的简单说明:https://blog.csdn.net/zawedx/article/details ...
- 启发式合并&线段树合并/分裂&treap合并&splay合并
启发式合并 有\(n\)个集合,每次让你合并两个集合,或询问一个集合中是否存在某个元素. 我们可以用平衡树/set维护集合. 对于合并两个\(A,B\),如果\(|A|<|B|\),那么 ...
- CF666E Forensic Examination [后缀自动机,线段树合并]
洛谷 Codeforces 思路 最初的想法:后缀数组+区间众数,似乎并不能过. 既然后缀数组不行,那就按照套路建出广义SAM,然后把\(S\)放在上面跑,得到以每个点结尾会到SAM上哪个节点. 询问 ...
- 【XSY1551】往事 广义后缀数组 线段树合并
题目大意 给你一颗trie树,令\(s_i\)为点\(i\)到根的路径上的字符组成的字符串.求\(max_{u\neq v}(LCP(s_u,s_v)+LCS(s_u,s_v))\) \(LCP=\) ...
- Codeforces.1051G.Distinctification(线段树合并 并查集)
题目链接 \(Description\) 给定\(n\)个数对\(A_i,B_i\).你可以进行任意次以下两种操作: 选择一个位置\(i\),令\(A_i=A_i+1\),花费\(B_i\).必须存在 ...
随机推荐
- Windows下Tomcat安装 + eclipse-Server配置
一.Tomcat安装 1.安装 解压安装包到自己的电脑上,建议不要解压到中文目录下 2.配置 网上有些安装教程提到需要配置CATALINA_HOME环境变量,此变量指向了tomcat 的目录,主要是为 ...
- C++11新特性之 Move semantics(移动语义)
https://blog.csdn.net/wangshubo1989/article/details/49748703 这篇讲到了vector的push_back的两种重载版本,左值版本和右值版本.
- OC学习篇之---@property和@synthesize的使用
在之前一片文章我们介绍了OC中的内存管理:http://blog.csdn.net/jiangwei0910410003/article/details/41924683,今天我们来介绍两个关键字的使 ...
- Tomcat内存问题解决办法
使用Java程序从数据库中查询大量的数据时出现异常:java.lang.OutOfMemoryError: Java heap space 在JVM中如果98%的时间是用于GC且可用的 Heap si ...
- vue的proxy和defineProperty区别
Object.defineProperty(obj,"name",{ set:function(val){ if(var==='lisi'){ console.log(" ...
- if语句里面continue和break的区别
break:结束整个循环体 continue:结束本次循环 代码说明: public static void main(String[] args) { int x=0; while(x++ < ...
- 在IIS7以上导出所有应用程序池的方法批量域名绑定(网站绑定)
在IIS7+上导出所有应用程序池的方法: %windir%/system32/inetsrv/appcmd list apppool /config /xml > c:/apppools.xml ...
- Newline required at end of file but not found
在启动vue项目中遇到 解决方法 在main.js文件的最后一行加一个空行就可以了
- TTL 与 CMOS
Frm: https://blog.csdn.net/qq_27745395/article/details/76687175 http://baijiahao.baidu.com/s?id=1598 ...
- 力扣算法题—149Max Points on a line
Given n points on a 2D plane, find the maximum number of points that lie on the same straight line. ...