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

时间:2023-01-08 23:10:13

3110: [Zjoi2013]K大数查询

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 6050  Solved: 2007
[Submit][Status][Discuss]

Description

有N个位置,M个操作。操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c
如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是多少。

Input

第一行N,M
接下来M行,每行形如1 a b c或2 a b c

Output

输出每个询问的结果

Sample Input

2 5
1 1 2 1
1 1 2 2
2 1 1 2
2 1 1 1
2 1 2 3

Sample Output

1
2
1

HINT

【样例说明】

第一个操作 后位置 1 的数只有 1 , 位置 2 的数也只有 1 。 第二个操作 后位置 1

的数有 1 、 2 ,位置 2 的数也有 1 、 2 。 第三次询问 位置 1 到位置 1 第 2 大的数 是

1 。 第四次询问 位置 1 到位置 1 第 1 大的数是 2 。 第五次询问 位置 1 到位置 2 第 3

大的数是 1 。‍

N,M<=50000,N,M<=50000

a<=b<=N

1操作中abs(c)<=N

2操作中c<=Maxlongint


本题做法好多啊,人太弱先写权值线段树套区间线段树吧

蒟蒻无脑总结:树套树是一种二维数据结构,用来解决二维的信息修改和询问

本题两个信息,一段区间中保存数,求第k大

显然用一个区间数据结构一个可求排名的数据结构,可以用权值线段树和区间线段树

区间线段树在外层,维护增加数的信息好难啊

权值线段树在外层,每个节点放一颗区间线段树维护这个节点权值在区间中出现次数

修改...修改就行了啊

查询,在权值线段树上二分答案,只不过用的是[a,b]中的出现次数

复杂度nlog^2n

参考别人写的,好诡异啊,明明有范围那么大还有负数,不离散化也能A

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
#define m ((l+r)>>1)
#define lc t[o].l
#define rc t[o].r
#define lson t[o].l,l,m
#define rson t[o].r,m+1,r
const int N=5e4+;
typedef long long ll;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
} int n,Q,op,ql,qr,v;
struct node{
int l,r;
ll sum,tag;
}t[N*];
int sz;
inline void merge(int o){
t[o].sum=t[lc].sum+t[rc].sum;
}
inline void paint(int &o,int l,int r,ll d){
if(!o) o=++sz;
t[o].sum+=(ll)(r-l+)*d;
t[o].tag+=d;
}
inline void pushDown(int o,int l,int r){
if(t[o].tag){
paint(lson,t[o].tag);
paint(rson,t[o].tag);
t[o].tag=;
}
}
void add(int &o,int l,int r,int ql,int qr,int d){
if(!o) o=++sz;
if(ql<=l&&r<=qr) paint(o,l,r,d);
else{
pushDown(o,l,r);
if(ql<=m) add(lson,ql,qr,d);
if(m<qr) add(rson,ql,qr,d);
merge(o);
}
}
ll query(int o,int l,int r,int ql,int qr){
if(!o) return ;
if(ql<=l&&r<=qr) return t[o].sum;
else{
pushDown(o,l,r);
ll ans=;
if(ql<=m) ans+=query(lson,ql,qr);
if(m<qr) ans+=query(rson,ql,qr);
return ans;
}
} int rt[N<<];
void add2(int o,int l,int r,int ql,int qr,int v){
add(rt[o],,n,ql,qr,);
if(l==r) return;
if(v<=m) add2(o<<,l,m,ql,qr,v);
else add2(o<<|,m+,r,ql,qr,v);
}
ll query2(int o,int l,int r,int ql,int qr,int k){
if(l==r) return l;
ll rsize=query(rt[o<<|],,n,ql,qr);
if(rsize>=k) return query2(o<<|,m+,r,ql,qr,k);
else return query2(o<<,l,m,ql,qr,k-rsize);
}
int main(){
//freopen("in.txt","r",stdin);
n=read();Q=read();
while(Q--){
op=read();ql=read();qr=read();v=read();
if(op==) add2(,,n,ql,qr,v);
else printf("%lld\n",query2(,,n,ql,qr,v));
}
}

BZOJ 3110: [Zjoi2013]K大数查询 [树套树]的更多相关文章

  1. BZOJ&period;3110&period;&lbrack;ZJOI2013&rsqb;K大数查询&lpar;整体二分 树状数组&sol;线段树&rpar;

    题目链接 BZOJ 洛谷 整体二分求的是第K小(利用树状数组).求第K大可以转为求第\(n-K+1\)小,但是这样好像得求一个\(n\). 注意到所有数的绝对值\(\leq N\),将所有数的大小关系 ...

  2. 树套树专题——bzoj 3110&colon; &lbrack;Zjoi2013&rsqb; K大数查询 &amp&semi;amp&semi; 3236 &lbrack;Ahoi2013&rsqb; 作业 题解

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

  3. bzoj 3110&colon; &lbrack;Zjoi2013&rsqb;K大数查询 树状数组套线段树

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

  4. BZOJ 3110&colon; &lbrack;Zjoi2013&rsqb;K大数查询&lpar; 树状数组套主席树 &rpar;

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

  5. BZOJ 3110&lpar;&lbrack;Zjoi2013&rsqb;K大数查询-区间第k大&lbrack;段修改&comma;在线&rsqb;-树状数组套函数式线段树&rpar;

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

  6. BZOJ 3110 &lbrack;Zjoi2013&rsqb;K大数查询(整体二分)

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 11654  Solved: 3505[Submit][St ...

  7. bzoj 3110 &lbrack;Zjoi2013&rsqb;K大数查询【树套树&vert;&vert;整体二分】

    树套树: 约等于是个暴力了.以区间线段树的方式开一棵权值线段树,在权值线段树的每一个点上以动态开点的方式开一棵区间线段树. 结果非常惨烈(时限20s) #include<iostream> ...

  8. BZOJ 3110 &lbrack;Zjoi2013&rsqb;K大数查询 ——树套树

    [题目分析] 外层区间线段树,内层是动态开点的权值线段树. SY神犇说树套树注重的是内外层的数据结构的选择问题,果然很重要啊. 动态开点的实现方法很好. [代码] #include <cstdi ...

  9. BZOJ 3110&colon; &lbrack;Zjoi2013&rsqb;K大数查询 &lbrack;整体二分&rsqb;

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

随机推荐

  1. vim 配置,我本机的配置&lbrack;windows&rsqb;

    set nocompatible source $VIMRUNTIME/vimrc_example.vim source $VIMRUNTIME/mswin.vim behave mswin set ...

  2. ios block中引用self

    __block __weak typeof(self) tmpSelf = self; ^(){ tmpSelf...... }

  3. ListView的深入学习

    ListView通常有两个职责: 将数据填充到布局 : 处理用户的点击选择操作 二.创建ListView需要3个元素 ListView的每一列的View View的数据或者图片 连接数据与ListVi ...

  4. JS中构造函数与函数

    //构造函数中,如果返回的是一个对 象,那么就保留原意. 如果返回的是非对象,比如数字.布尔和字符串,那么就返回 this,如果没有 return 语句,那么也返回this. var myFun1 = ...

  5. 随笔之Android平台上的进程调度探讨

    http://blog.csdn.net/innost/article/details/6940136 随笔之Android平台上的进程调度探讨 一由来 最近在翻阅MediaProvider的时候,突 ...

  6. Linux下JDK安装位置

    新手在Linux上安装JDK时,不知道应该将JDK安装在哪比较合适.首先简要了解下Linux中部分目录的作用. /bin---用来贮存用户命令./usr/bin 也被用来贮存用户命令.  /sbin- ...

  7. PYTHONE的WHILE,BREAK&comma;CONTINUE示例

    简短示例: while True: s = raw_input('Enter something : ') if s == 'quit': break if len(s) < 3: print ...

  8. hdu 4919 Exclusive or

    Exclusive or Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) T ...

  9. pip install -r requirements&period;txt 安装mysqldb失败 解决方案

    在pip.log中出现sh: 1: mysql_config: not found等一坨报错,因为没有安装另一个包: 只要原因是没有安装:libmysqlclient-dev sudo apt-get ...

  10. VBS列出windows更新列表

    Set objSession = CreateObject("Microsoft.Update.Session") Set objSearcher = objSession.Cre ...