BZOJ 3065 带插入区间K小值 (替罪羊树套线段树)
毒瘤题.参考抄自博客:hzwer
- 第一次写替罪羊树,完全是照着题解写的,发现这玩意儿好强啊,不用旋转每次都重构还能nlognnlognnlogn.
- 还有外面二分和里面线段树的值域一样,那么r = mid就相当于线段树往左儿子走,l = mid + 1就相当于线段树往右儿子走,神了.(这样的话就有点像决策始终一致的整体二分了口胡).
CODE
TIP:TIP:TIP: 参考博客的代码92行应该是小写的"r>L+1"…不过都能过.
#include <vector>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
char cb[1<<15],*cs=cb,*ct=cb;
#define getc() (cs==ct&&(ct=(cs=cb)+fread(cb,1,1<<15,stdin),cs==ct)?0:*cs++)
template<class T>inline void read(T &res) {
char ch; for(;!isdigit(ch=getc()););
for(res=ch-'0';isdigit(ch=getc());res=res*10+ch-'0');
}
const int MAXN = 70005; //N+Q最大值
const int MAXV = 70000; //权值最大值
const int MAXNN = 10000005; //线段树结点
const double alpha = 0.75;
int n, q, root, tmp, sz;
int v[MAXN], num[MAXN], rt[MAXN], ch[MAXN][2];
struct seg{ int ls, rs, sum; }a[MAXNN];
vector<int> rec, p, t;
inline int newnode() {
if(!rec.size()) return ++sz;
int o = rec.back(); rec.pop_back();
return o;
}
void reclaim(int &i) { //seg
if(!i) return;
rec.push_back(i);
reclaim(a[i].ls), reclaim(a[i].rs);
a[i].sum = 0;
i = 0; //注意顺序!
}
void del(int &i) { //sgt
if(!i) return; reclaim(rt[i]);
del(ch[i][0]); p.push_back(i); del(ch[i][1]);
i = 0;
}
void insert(int &i, int l, int r, int x, int val) {
if(!i) i = newnode();
if(l == r) { a[i].sum += val; return; }
int mid = (l + r) >> 1;
if(x <= mid) insert(a[i].ls, l, mid, x, val);
else insert(a[i].rs, mid+1, r, x, val);
a[i].sum = a[a[i].ls].sum + a[a[i].rs].sum;
if(!a[i].sum) reclaim(i);
}
void build(int &i, int l, int r) {
if(l > r) return;
if(l == r) { i = num[l]; insert(rt[i], 0, MAXV, v[i], 1); return; }
int mid = (l + r) >> 1; i = num[mid];
build(ch[i][0], l, mid-1);
build(ch[i][1], mid+1, r);
for(int j = l; j <= r; ++j)
insert(rt[i], 0, MAXV, v[num[j]], 1);
}
inline void rebuild(int &i) {
del(i); int siz = p.size();
for(int l = 1; l <= siz; ++l) num[l] = p[l-1];
build(i, 1, siz); p.clear();
}
int Modify(int i, int x, int val) {
insert(rt[i], 0, MAXV, val, 1);
int ret, L = a[rt[ch[i][0]]].sum;
if(L + 1 == x) { ret = v[i]; v[i] = val; }
else if(L >= x) ret = Modify(ch[i][0], x, val);
else ret = Modify(ch[i][1], x-L-1, val);
insert(rt[i], 0, MAXV, ret, -1);
return ret;
}
void Insert(int &i, int x, int val) {
if(!i) { i = ++n; insert(rt[i], 0, MAXV, val, 1); v[i] = val; return; }
insert(rt[i], 0, MAXV, val, 1);
int L = a[rt[ch[i][0]]].sum;
if(L >= x) Insert(ch[i][0], x, val);
else Insert(ch[i][1], x-L-1, val);
if(a[rt[i]].sum * alpha > max(a[rt[ch[i][0]]].sum, a[rt[ch[i][1]]].sum)) {
if(tmp) {
if(ch[i][0] == tmp) rebuild(ch[i][0]);
else rebuild(ch[i][1]); tmp = 0;
}
}
else tmp = i;
}
void query(int i, int l, int r) {
int L = a[rt[ch[i][0]]].sum, Size = a[rt[i]].sum;
if(l == 1 && r == Size) { t.push_back(rt[i]); return; }
if(l <= L+1 && r >= L+1) p.push_back(v[i]);
if(r <= L) query(ch[i][0], l, r);
else if(l > L+1) query(ch[i][1], l-L-1, r-L-1);
else {
if(l <= L) query(ch[i][0], l, L);
if(r > L+1) query(ch[i][1], 1, r-L-1);
}
}
inline int Query(int L, int R, int K) {
query(root, L, R);
int l = 0, r = 70000, sizt = t.size(), sizp = p.size();
while(l < r) {
int mid = (l + r) >> 1, tot = 0;
for(int i = 0; i < sizt; ++i) tot += a[a[t[i]].ls].sum;
for(int i = 0; i < sizp; ++i) if(p[i] >= l && p[i] <= mid) ++tot;
if(K <= tot) {
for(int i = 0; i < sizt; ++i) t[i] = a[t[i]].ls;
r = mid;
}
else {
for(int i = 0; i < sizt; ++i) t[i] = a[t[i]].rs;
l = mid+1; K -= tot;
}
}
t.clear(); p.clear(); return l;
}
int main () {
read(n);
for(int i = 1; i <= n; ++i)
read(v[i]), num[i] = i;
build(root, 1, n);
read(q);
char s[2]; int x, y, K, lastans = 0;
while(q--) {
while(!isalpha(s[0]=getc()));
while(isalpha(s[1]=getc()));
read(x), read(y);
x ^= lastans, y ^= lastans;
if(s[0] == 'Q') read(K), K ^= lastans, printf("%d\n", lastans=Query(x, y, K));
else if(s[0] == 'M') Modify(root, x, y);
else {
tmp = 0; Insert(root, x-1, y); if(tmp) rebuild(root);
}
}
return 0;
}
BZOJ 3065 带插入区间K小值 (替罪羊树套线段树)的更多相关文章
- bzoj 3065: 带插入区间K小值 替罪羊树 && AC300
3065: 带插入区间K小值 Time Limit: 60 Sec Memory Limit: 512 MBSubmit: 1062 Solved: 253[Submit][Status] Des ...
- 【题解】BZOJ 3065: 带插入区间K小值——替罪羊树套线段树
题目传送门 题解 orz vfk的题解 3065: 带插入区间K小值 系列题解 一 二 三 四 惨 一开始用了一种空间常数很大的方法,每次重构的时候merge两颗线段树,然后无限RE(其实是MLE). ...
- BZOJ 3065 带插入区间K小值(sag套线段树)
3065: 带插入区间K小值 Time Limit: 60 Sec Memory Limit: 512 MBSubmit: 4696 Solved: 1527[Submit][Status][Di ...
- bzoj 3065: 带插入区间K小值(分块)
Description 从前有n只跳蚤排成一行做早操,每只跳蚤都有自己的一个弹跳力a[i].跳蚤国王看着这些跳蚤国欣欣向荣的情景,感到非常高兴.这时跳蚤国王决定理性愉悦一下,查询区间k小值.他每次向它 ...
- BZOJ 3065 带插入区间K小值
http://www.lydsy.com/JudgeOnline/problem.php?id=3065 思路:替罪羊树套权值线段树. 当替罪羊树某个子树大于某个比利(比例)时就暴力重构,本题时间复杂 ...
- 3065: 带插入区间K小值_树套树_替罪羊树_权值线段树
经过周六一天,周一3个小时的晚自习,周二2个小时的疯狂debug,终于凭借自己切掉了这道树套树题. Code: #include <cstdio> #include <algorit ...
- 【BZOJ3065】带插入区间K小值 替罪羊树+权值线段树
[BZOJ3065]带插入区间K小值 Description 从前有n只跳蚤排成一行做早操,每只跳蚤都有自己的一个弹跳力a[i].跳蚤国王看着这些跳蚤国欣欣向荣的情景,感到非常高兴.这时跳蚤国王决定理 ...
- 【BZOJ】3065: 带插入区间K小值
http://www.lydsy.com/JudgeOnline/problem.php?id=3065 题意:带插入.修改的区间k小值在线查询.(原序列n<=35000, 询问<=175 ...
- 【bzoj3065】带插入区间K小值 替罪羊树套权值线段树
题目描述 从前有n只跳蚤排成一行做早操,每只跳蚤都有自己的一个弹跳力a[i].跳蚤国王看着这些跳蚤国欣欣向荣的情景,感到非常高兴.这时跳蚤国王决定理性愉悦一下,查询区间k小值.他每次向它的随从伏特提出 ...
随机推荐
- 什么是云解析DNS?
产品概述 云解析DNS(Alibaba Cloud DNS)是一种安全.快速.稳定.可扩展的权威DNS服务,云解析DNS为企业和开发者将易于管理识别的域名转换为计算机用于互连通信的数字IP地址,从而将 ...
- 南昌网络赛J. Distance on the tree 树链剖分
Distance on the tree 题目链接 https://nanti.jisuanke.com/t/38229 Describe DSM(Data Structure Master) onc ...
- 20190724-Python网络数据采集/第 2 章 复杂HTML解析-导航树/正则表达式
1. 导航树 经典的HTML树状结构 直接看下面的代码示例:(注意目标网页的标签大小写等细节,易出bug) from urllib.request import urlopen from bs4 im ...
- 简单使用setup.py来安装Python项目
最近做个一个项目需要用到setup.py 这个构建工具来进行项目的便捷安装,把搜集到的一些资料加上个人理解整理成文章,如有错误的地方请各位大佬及时指出,小弟马上修改,下面正式进入setup.py的描述 ...
- 【思维】ABC
题目描述 You are given a string s consisting of A, B and C.Snuke wants to perform the following operatio ...
- 教你如何进行移动端APP测试
1.安全测试(权限) 1)软件权限:其中包括发送信息,拨打电话,链接网络,访问手机信息,联系人信息等等 2)数据在本地的存储.传输等 3)执行某些操作时导致的输入有效性验证.授权.数据加密等方面 4) ...
- Springboot+mybatis+druid 配置多数据源
项目结构 application.yml配置文件 spring: application: name: service datasource: primary: jdbc-url: jdbc:orac ...
- php实现拼图滑块验证的思考及部分实现
实现拼图滑块验证,我觉得其中比较关键的一点就是裁剪图片,最起码需要裁剪出下面两张图的样子 底图 滑块图 一张底图和一张滑块图,其中底图实现起来比较简单可以使用添加水印的方式直接将一张拼图形状的半透明图 ...
- js同步和异步
JavaScript语言的一大特点就是单线程,也就是说,同一个时间只能做一件事. JavaScript的单线程,与它的用途有关.作为浏览器脚本语言,JavaScript的主要用途是与用户互动,以及操作 ...
- Java优化高性能高并发+高并发程序设计视频教程
转自:https://www.cnblogs.com/ajianku/p/10236573.html 第1章 课程介绍及项目框架搭建1-1 Java高并发商城秒杀优化导学1-2 项目环境搭建(Ecli ...