题目描述

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

输入

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

输出

输出每个询问的结果

样例输入

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

样例输出

1
2
1

提示

【样例说明】

第一个操作 后位置 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

显然一棵线段树或平衡树无法维护这道题的操作,那么考虑树套树。

一开始可能会想将维护序列的线段树放在外层,然后将维护序列每个点各种权值的权值线段树放在内层。

这样修改操作就是外层线段树区间修改,内层线段树单点修改,很好操作。

但你发现无法完成询问操作,如果用二分答案来解决的话时间复杂度就是O(nlogn^3)的了。

既然询问第k大,那么显然权值线段树对于这种操作更为擅长,我们将权值线段树放在外层,将维护序列的线段树放在内层。

这样修改就变成了外层线段树单点修改,内层线段树区间修改,同样很好操作。

对于询问,我们每查询到权值线段树的一个节点先在这个点左子节点的内层线段树上区间查找,即找出序列上询问区间中权值小于某个值的数量,再与k判断来决定往左走还是往右走。

#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<bitset>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
using namespace std;
int n,m;
int opt;
int cnt;
int a,b;
ll c;
ll s[20000010];
int root[2000010];
ll sum[20000010];
int ls[20000010];
int rs[20000010];
inline void pushup(int rt)
{
sum[rt]=sum[ls[rt]]+sum[rs[rt]];
}
inline void pushdown(int rt,int l,int r)
{
if(s[rt])
{
int mid=(l+r)>>1;
if(!ls[rt])
{
ls[rt]=++cnt;
}
if(!rs[rt])
{
rs[rt]=++cnt;
}
s[ls[rt]]+=s[rt];
s[rs[rt]]+=s[rt];
sum[ls[rt]]+=s[rt]*(mid-l+1);
sum[rs[rt]]+=s[rt]*(r-mid);
s[rt]=0;
}
}
inline void change(int &rt,int l,int r,int L,int R)
{
if(!rt)
{
rt=++cnt;
}
if(L<=l&&r<=R)
{
sum[rt]+=(r-l+1);
s[rt]++;
return ;
}
pushdown(rt,l,r);
int mid=(l+r)>>1;
if(L<=mid)
{
change(ls[rt],l,mid,L,R);
}
if(R>mid)
{
change(rs[rt],mid+1,r,L,R);
}
pushup(rt);
}
inline ll find(int rt,int l,int r,int L,int R)
{
if(!rt)
{
return 0;
}
if(L<=l&&r<=R)
{
return sum[rt];
}
pushdown(rt,l,r);
int mid=(l+r)>>1;
ll res=0;
if(L<=mid)
{
res+=find(ls[rt],l,mid,L,R);
}
if(R>mid)
{
res+=find(rs[rt],mid+1,r,L,R);
}
return res;
}
inline void insert(int rt,int l,int r,int k,int L,int R)
{
change(root[rt],1,n,L,R);
if(l==r)
{
return ;
}
int mid=(l+r)>>1;
if(k<=mid)
{
insert(rt<<1,l,mid,k,L,R);
}
else
{
insert(rt<<1|1,mid+1,r,k,L,R);
}
}
inline int query(int rt,int l,int r,ll k,int L,int R)
{
if(l==r)
{
return l;
}
int mid=(l+r)>>1;
ll res=find(root[rt<<1|1],1,n,L,R);
if(res<k)
{
return query(rt<<1,l,mid,k-res,L,R);
}
else
{
return query(rt<<1|1,mid+1,r,k,L,R);
}
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++)
{
scanf("%d",&opt);
scanf("%d%d%lld",&a,&b,&c);
if(a>b)
{
swap(a,b);
}
if(opt==1)
{
c=(int)c;
insert(1,0,2*n,c+n,a,b);
}
else
{
printf("%d\n",query(1,0,2*n,c,a,b)-n);
}
}
}

BZOJ3110[Zjoi2013]K大数查询——权值线段树套线段树的更多相关文章

  1. 【bzoj3110】[Zjoi2013]K大数查询 权值线段树套区间线段树

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

  2. 洛谷P3332 [ZJOI2013]K大数查询 权值线段树套区间线段树_标记永久化

    Code: #include <cstdio> #include <algorithm> #include <string> #include <cstrin ...

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

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

  4. BZOJ3110[Zjoi2013]K大数查询(树状数组+整体二分)

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

  5. bzoj3110: [Zjoi2013]K大数查询 【树套树,标记永久化】

    //========================== 蒟蒻Macaulish:http://www.cnblogs.com/Macaulish/  转载要声明! //=============== ...

  6. BZOJ3110 [Zjoi2013]K大数查询 树套树 线段树 整体二分 树状数组

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ3110 题意概括 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位 ...

  7. bzoj3110 [Zjoi2013]K大数查询——线段树套线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 外层权值线段树套内层区间线段树: 之所以外层权值内层区间,是因为区间线段树需要标记下传 ...

  8. BZOJ3110: [Zjoi2013]K大数查询

    喜闻乐见的简单树套树= =第一维按权值建树状数组,第二维按下标建动态开点线段树,修改相当于第二维区间加,查询在树状数组上二分,比一般的线段树还短= =可惜并不能跑过整体二分= =另外bzoj上的数据有 ...

  9. [BZOJ3110] [Zjoi2013] K大数查询 (树套树)

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

随机推荐

  1. C语言程序设计II—第八周教学

    第八周教学总结(15/4-21/4) 教学内容 本周的教学内容为: 8.4 电码加密 知识点:指针与字符串,重难点:字符指针与字符串的关联和区别: 8.5 任意个整数求和 知识点:动态内存分配的概念和 ...

  2. jdk_1_8_1

    JAVA_HOME=/usr/local/java/jdk1.8.0_181 PATH=$JAVA_HOME/bin:$PATH JAVA_BINDIR=/usr/local/java/jdk1.8. ...

  3. 开放的dae模型

    从网上看到了这段代码,就Copy过来了. 其实面对dae这种开放的模型格式,我们可以做很多事情,就像通常的XML文件一样. //------------------------------------ ...

  4. Oracle 在函数或存储过程中执行一条插入语句并返回主键ID值

    有时,我们需要往一张表插入一条记录,同时返回主键ID值. 假定主键ID的值都是通过对应表的SEQUENCE来获得,然后进行ID赋值 这里有几种情况需要注意: 1)如果建表语句含有主键ID的触发器,通过 ...

  5. 快速排序的php实现

    再来一个非常高级的排序算法,快速排序...这个算法是很高效的. 快速排序的思路是,找到一个分割点(中枢点 默认是列表第一个值),把原列表分隔成两部分,在分割点左侧的是都比它小的,在它右侧的是都比它大的 ...

  6. ASP.NET Core 中 HttpContext 详解与使用 | Microsoft.AspNetCore.Http 详解 (转载)

    “传导体” HttpContext 要理解 HttpContext 是干嘛的,首先,看图 图一 内网访问程序 图二 反向代理访问程序 ASP.NET Core 程序中,Kestrel 是一个基于 li ...

  7. CF [2016-2017 ACM-ICPC CHINA-Final][GYM 101194 H] Great Cells

    很久以前做的一道思博题了,今天来补一补. 大致题意:在一个\(n*m\)的矩阵内填整数,数字在\([1,k]\)范围内.矩阵中某格的数为great number当且仅当与它同行同列的数字都严格比它小. ...

  8. 数列分块入门九题(二):LOJ6280~6282

    Preface 个人感觉这中间的三题是最水的没有之一 数列分块入门 4--区间加法,区间求和 这个也是很多数据结构完爆的题目线段树入门题,但是练分块我们就要写吗 修改还是与之前类似,只不过我们要维护每 ...

  9. Jlink使用技巧之虚拟串口功能

    前言 串口调试是单片机开发过程必不可少的一个功能,一般是使用一个UART-TTL的串口模块来实现串口的功能,其实下载调试使用的Jlink仿真器也可以实现串口调试的功能,本篇文章将介绍如何使用Jlink ...

  10. .NetCore实践篇:成功解决分布式监控ZipKin聚合依赖问题(三)

    前言 读本篇文章之前,可以先读前两篇文章.为了照顾没看过的朋友,我也会稍作复习. 思考大纲: .Net架构篇:思考如何设计一款实用的分布式监控系统? 实践篇一:.NetCore实践篇:分布式监控客户端 ...