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\).必须存在 ...
随机推荐
- Mybatis基于XML配置SQL映射器(三)
Mybatis之动态SQL mybatis 的动态sql语句是基于OGNL表达式的.可以方便的在 sql 语句中实现某些逻辑. 总体说来mybatis 动态SQL 语句主要有以下几类: if choo ...
- POJ 2240 Arbitrage (spfa判环)
Arbitrage Arbitrage is the use of discrepancies in currency exchange rates to transform one unit of ...
- JavaScript 的 API设计原则
一.接口的流畅性 好的接口是流畅易懂的,他主要体现如下几个方面: 1.简单 操作某个元素的css属性,下面是原生的方法: document.querySelectorAll('#id').style. ...
- linux查看java jdk jre安装路径和设置环境变量
一. 查看java jdk安装路径和设置环境变量 windows: set java_home:查看JDK安装路径 java -version:查看JDK版本 linux: whereis java ...
- windows7+tomcat7+nginx1.11.3 +memcached
测试的环境是windows7+tomcat7+nginx1.11.3 +memcached 安装方法网上很多就不多说了. 1.session共享需要这几个jar 包 下载地址 http://down ...
- shell从字符串中提取子串(正则表达式)
通过试验,可以通过grep.sed两种方式实现. 假设需要提取libgcc-4.8.5-4.h5.x86_64.rpm中的版本号. grep echo "libgcc-4.8.5-4.h5. ...
- selenium2-java环境搭建 示例为chrome浏览器
首先,安装并配置JDK,安装eclipse,安装firefox和chrome.下载selenium语言的JAVA库文件,下载地址为,如果打不开,则需要翻墙:http://www.seleniumhq. ...
- Spring事物的传播
spring的事物对于同一个类内部调用是不会生效的. 比如一个ServiceA,里面有个方法x()和y().其中x没有配置事物,而y配置的有实物.如果是一个没有事物的ServiceB调用了Servic ...
- Linux基础-命令概述
概述 很多人可能在电视或电影中看到过类似的场景,黑客面对一个黑色的屏幕,上面飘着密密麻麻的字符,梆梆一顿敲,就完成了窃取资料的任务,是不是很帅!我们作为一个开发者, 即使不为了成为上述的人, 也需要会 ...
- python学习笔记:模块——自定义模块的3种导入方式
一.定义 模块就是用一堆的代码实现了一些功能的代码的集合,通常一个或者多个函数写在一个.py文件里,而如果有些功能实现起来很复杂,那么就需要创建n个.py文件,这n个.py文件的集合就是模块.如果不懂 ...