题意:戳这里

思路:可以用cdq分治(很明显这种模型妹纸分治法很解决)。。不过为了学习树套树特地写了一下。。

所谓的树套树也第一层(最外层)普通的维护的是一个node,而树套树维护的是一个数据结构(一棵树)。。

树套树一般可以解决2维模型。。1维的话也就是普通的数据结构了。

比如poi07 的mokia其实就是一个2为线段树,不够空间不够所以必须写成树套树。。

本题的话如把权值看成一维,本来位置看成1维,那么其实也是2维模型。

插入就等价于每次在一条x=c 横线的[a,b]之间每个位置都插入1遍

查询等价于求第k大的在哪条横线上。。

对于这一题的话,可以如下:

首先最外层维护的是权值构成的线段树,

而对于每个权值,又对应着一棵线段树,不过这个线段树是下标线段树

由于空间有限,所以有用到才动态分配内存。。

然后每次插入的话在最外层包括value=c的logn段里面都插入,

查询的每次二分,左边太小右边找,正好从外层线段树从上到下。。

时间复杂度O(mlog2n)

code:

 #include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<string>
#include<map>
#include<set>
#include<vector>
#include<queue>
#include<stack>
#define M0(x) memset(x, 0, sizeof(x))
using namespace std;
#define lson lc[rt], l, m
#define rson rc[rt], m+1, r
const int N = , M = ;
int rt[N<<], sum[M], lc[M], rc[M], lz[M];
int n, m, L, R, cnt, c; inline void push_up(const int& rt){
sum[rt] = sum[lc[rt]] + sum[rc[rt]];
} inline void push_down(const int &rt, const int& l, const int &r){
if (lz[rt]){
if (!lc[rt]) lc[rt] = ++cnt;
if (!rc[rt]) rc[rt] = ++cnt;
sum[lc[rt]] += lz[rt] * ((r-l+)>>), sum[rc[rt]] += lz[rt] * ((r-l+)>>);
lz[lc[rt]] += lz[rt], lz[rc[rt]] += lz[rt];
lz[rt] = ;
}
} int query(const int& rt,const int& l, const int& r){
if (!rt) return ;
if (L <= l && r <= R) return sum[rt];
int m = (l + r) >> , tmp1 = , tmp2 = ;
push_down(rt, l, r);
if (L <= m) tmp1 = query(lson);
if (R > m) tmp2 = query(rson);
return tmp1 + tmp2 + (min(R, r) - max(l, L) + ) * lz[rt];
} int query(int k){
int l = , r = n, mid, cur = , tmp;
while (l <= r){
if (l == r) return l;
mid = (l + r) >> ;
tmp = query(rt[cur<<], , n);
if (tmp >= k) r = mid, cur <<= ;
else l = mid + , k -= tmp, cur = cur<<|;
}
return l;
} void update(int &rt,const int& l,const int& r){
if (!rt) rt = ++cnt;
if (L <= l && r <= R){
sum[rt] += (r - l + ), ++lz[rt];
return;
}
int m = (l + r) >> ;
push_down(rt, l, r);
if (L <= m) update(lson);
if (R > m) update(rson);
push_up(rt);
} void insert(const int& c){
int l = , r = n, cur = , mid;
while (l <= r){
update(rt[cur], , n);
if (l == r) break;
mid = (l + r) >> ;
if (c <= mid) cur<<= , r = mid;
else l = mid + , cur = cur<<|;
}
} void solve(){
cnt = ;
int op;
while (m--){
scanf("%d%d%d%d", &op, &L, &R, &c);
if (op==) c = n - c + , insert(c);
else printf("%d\n", n - query(c) + );
}
} int main(){
freopen("a.in", "r", stdin);
freopen("a.out", "w", stdout);
while (scanf("%d%d", &n, &m) != EOF){
solve();
}
return ;
}

code

 #include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<string>
#include<map>
#include<set>
#include<vector>
#include<queue>
#include<stack>
#define M0(x) memset(x, 0, sizeof(x))
using namespace std;
#define lson lc[rt], l, m
#define rson rc[rt], m+1, r
const int N = , M = ;
int rt[N<<], sum[M], lc[M], rc[M], lz[M];
int n, m, L, R, cnt, c; int query(const int& rt,const int& l, const int& r){
if (!rt) return ;
if (L <= l && r <= R) return sum[rt];
int m = (l + r) >> , tmp1 = , tmp2 = ;
if (L <= m) tmp1 = query(lson);
if (R > m) tmp2 = query(rson);
return tmp1 + tmp2 + (min(R, r) - max(l, L) + ) * lz[rt];
} int query(int k){
int l = , r = n, mid, cur = , tmp;
while (l <= r){
if (l == r) return l;
mid = (l + r) >> , tmp = query(rt[cur<<], , n);
if (tmp >= k) r = mid, cur <<= ;
else l = mid + , k -= tmp, cur = cur<<|;
}
return l;
} void update(int &rt,const int& l,const int& r){
if (!rt) rt = ++cnt;
if (L <= l && r <= R){
sum[rt] += (r - l + ), ++lz[rt];
return;
}
int m = (l + r) >> ;
if (L <= m) update(lson);
if (R > m) update(rson);
sum[rt] = sum[lc[rt]] + sum[rc[rt]] + lz[rt] * (r - l + );
} void insert(const int& c){
int l = , r = n, cur = , mid;
while (l <= r){
update(rt[cur], , n);
if (l == r) break;
mid = (l + r) >> ;
if (c <= mid) cur<<= , r = mid;
else l = mid + , cur = cur<<|;
}
} void solve(){
cnt = ;
int op;
while (m--){
scanf("%d%d%d%d", &op, &L, &R, &c);
if (op==) c = n - c + , insert(c);
else printf("%d\n", n - query(c) + );
}
} int main(){
freopen("a.in", "r", stdin);
freopen("a.out", "w", stdout);
while (scanf("%d%d", &n, &m) != EOF){
solve();
}
return ;
}

第一个lazy直接下放慢成狗。。学习了一下优美姿势快了不少。。

bzoj 3110的更多相关文章

  1. [BZOJ 3110] [Zjoi2013] K大数查询 【树套树】

    题目链接: BZOJ - 3110 题目分析 这道题是一道树套树的典型题目,我们使用线段树套线段树,一层是区间线段树,一层是权值线段树.一般的思路是外层用区间线段树,内层用权值线段树,但是这样貌似会很 ...

  2. BZOJ 3110 K大数查询 | 整体二分

    BZOJ 3110 K大数查询 题面 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个 ...

  3. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

  4. BZOJ 3110: [Zjoi2013]K大数查询 [树套树]

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 6050  Solved: 2007[Submit][Sta ...

  5. BZOJ 3110 k大数查询 & 树套树

    题意: 有n个位置,每个位置可以看做一个集合,现在要求你实现一个数据结构支持以下功能: 1:在a-b的集合中插入一个数 2:询问a-b集合中所有元素的第k大. SOL: 调得火大! 李建说数据结构题能 ...

  6. 树套树专题——bzoj 3110: [Zjoi2013] K大数查询 &amp; 3236 [Ahoi2013] 作业 题解

    [原题1] 3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MB Submit: 978  Solved: 476 Descri ...

  7. bzoj 3110: [Zjoi2013]K大数查询 树状数组套线段树

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1384  Solved: 629[Submit][Stat ...

  8. BZOJ 3110: [Zjoi2013]K大数查询( 树状数组套主席树 )

    BIT+(可持久化)权值线段树, 用到了BIT的差分技巧. 时间复杂度O(Nlog^2(N)) ---------------------------------------------------- ...

  9. BZOJ 3110([Zjoi2013]K大数查询-区间第k大[段修改,在线]-树状数组套函数式线段树)

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec   Memory Limit: 512 MB Submit: 418   Solved: 235 [ Submit][ ...

  10. BZOJ 3110:[Zjoi2013]K大数查询(整体二分)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3110 题意:-- 思路:其实和之前POJ那道题差不多,只不过是换成区间更新,而且是第k大不是第k小, ...

随机推荐

  1. 关于兼容性——百分比对于IE浏览器的影响

    之前为了适应页面放大或者缩小,很多宽度都是写的百分比,后来发现有产生换行现象 总结一下: 如果页面是满占位的,我想百分比在谷歌.火狐都应该没什么问题,在IE 就会产生换行  因为IE浏览器存在一个四舍 ...

  2. webpack 的简单使用

    p.p2 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #323333 } p. ...

  3. TreeMap的使用

    Map<Integer, String> map = new TreeMap<Integer, String>();//TreeMap本身具有排序功能(默认按键升序排序) ma ...

  4. yarn container启动失败

    在yarn资源管理的集群上运行spark程序,无法读取的数据多与少,都会报这个错误,但是其他程序在集群上能够正常运行. 16/11/14 00:13:44 WARN cluster.YarnSched ...

  5. c++父类和子类转化致命的代码错误

    最近在工作中,出现了严重的代码错误,对象的基类和子类的继承,代码大致如下: class A { }; class B : public A { } void main() { A* a;(用于子类对象 ...

  6. MongoDB的TruncationException异常解决方法

    近期由于DB4O的一些无解的BUG,导致现行的项目查询效率底下,于是愤而迁移到了MongoDB. MongoDB虽然比DB4O的用户要多一些,但是文档依然极度匮乏,遇到问题不是那么容易就能搜到解决办法 ...

  7. 使用angularJS遇见的一些问题的解决方案

    1. angularJS的$http.post请求,SpringMVC后台接收不到参数值的解决方案 问题一般为:400 Required String parameter 'rPassword' is ...

  8. 在Mac上搭建本地Apache服务器一些注意点

    一般在开发ios程序中,我们需要使用到和服务器的交互操作. 一般我们在Mac上使用Apache来搭建服务器.数据库采用MySQL.在Mac中Apache是自带的.所有,我们可以不需要额外的去Apple ...

  9. Linux C++ 开发常用工具,常用指令工作手册

    vim常用: :set nu显示行数 :set mouse=a 鼠标滑动屏幕,:set ic :set noic 忽略不忽略大小写 /word_to_search\c \c表示忽略大小写 c小写忽略, ...

  10. hashcode深入理解

    带着问题去理解: 1. Object类HashCode方法是如何实现的,和String类有什么区别? 2.HashCode和Equals之间的关系? 一.hashCode作用 hashCode方法返回 ...