bzoj3110 [Zjoi2013]K大数查询——线段树套线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110
外层权值线段树套内层区间线段树;
之所以外层权值内层区间,是因为区间线段树需要标记下传,所以写在内层比较方便;
然而空间太大了,所以动态开点,大约每个外层线段树的点上有 logn 个内层线段树点;
最开始写的不知为何很快就WA:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const maxn=5e4+,maxm=maxn*;
int n,m,rt[maxn<<],ls[maxm],rs[maxm],cnt;
int a[maxn],b[maxn],tp[maxn],c[maxn],tmp[maxn],tot;
ll sum[maxm],lzy[maxm];//ll lzy 防止计算时爆int
void pushdown(int x,int l,int r)
{
if(!lzy[x])return;
if(!ls[x])ls[x]=++cnt;
if(!rs[x])rs[x]=++cnt;
int mid=((l+r)>>);
sum[ls[x]]+=(mid-l+)*lzy[x]; sum[rs[x]]+=(r-mid)*lzy[x];
lzy[ls[x]]+=lzy[x]; lzy[rs[x]]+=lzy[x];
lzy[x]=;
}
void add(int &x,int l,int r,int L,int R)
{
if(!x)x=++cnt;
if(l>=L&&r<=R){sum[x]+=(r-l+); lzy[x]++; return;}
// pushdown(x,l,r);
int mid=((l+r)>>);
if(mid>=L)add(ls[x],l,mid,L,R);
if(mid<R)add(rs[x],mid+,r,L,R);
sum[x]=sum[ls[x]]+sum[rs[x]];
}
void insert(int x,int l,int r,int tl,int tr,int c)
{
add(rt[x],,n,tl,tr);
if(l==r)return;//
int mid=((l+r)>>);//权值区间
if(c<=mid)insert(x<<,l,mid,tl,tr,c);
else insert(x<<|,mid+,r,tl,tr,c);
}
ll ask(int x,int l,int r,int L,int R)
{
if(!x)return ;//
if(l>=L&&r<=R)return sum[x];
pushdown(x,l,r);
int mid=((l+r)>>); ll ret=;
// if(mid>=L)ret+=ask(x<<1,l,mid,L,R);
// if(mid<R)ret+=ask(x<<1|1,mid+1,r,L,R);
if(mid>=L)ret+=ask(ls[x],l,mid,L,R);
if(mid<R)ret+=ask(rs[x],mid+,r,L,R);//别写串了!
return ret;
}
int query(int x,int l,int r,int L,int R,int k)
{
if(l==r)return l;
ll tmp=ask(rt[x<<|],,n,L,R),mid=((l+r)>>);//查询第k大
if(tmp>=k)return query(x<<|,mid+,r,L,R,k);//
else return query(x<<,l,mid,L,R,k-tmp);//
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d%d%d",&tp[i],&a[i],&b[i],&c[i]);
if(tp[i]==)tmp[++tot]=c[i];
}
sort(tmp+,tmp+n+); tot=unique(tmp+,tmp+n+)-tmp-;
for(int i=;i<=m;i++)
{
if(tp[i]==)
{
int tt=lower_bound(tmp+,tmp+tot+,c[i])-tmp;
insert(,,tot,a[i],b[i],tt);
}
else printf("%d\n",tmp[query(,,tot,a[i],b[i],c[i])]);
}
return ;
}
囧
不会改了,所以模仿别人写了个标记永久化,然后WA惨;
因为不太熟悉标记永久化,没注意到更新以及查询时要注意不能重复,改了半天,终于好了...
学到了点标记永久化的细节。
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const maxn=5e4+,maxm=maxn*;
int n,m,rt[maxn<<],ls[maxm],rs[maxm],cnt;
int a[maxn],b[maxn],tp[maxn],c[maxn],tmp[maxn],tot,L,R;
ll sum[maxm],lzy[maxm];//ll lzy 防止计算时爆int
//void pushdown(int x,int l,int r)
//{
// if(!lzy[x])return;
// if(!ls[x])ls[x]=++cnt;
// if(!rs[x])rs[x]=++cnt;
// int mid=((l+r)>>1);
// sum[ls[x]]+=(mid-l+1)*lzy[x]; sum[rs[x]]+=(r-mid)*lzy[x];
// lzy[ls[x]]+=lzy[x]; lzy[rs[x]]+=lzy[x];
// lzy[x]=0;
//}
void add(int &x,int l,int r,int L,int R)
{
if(!x)x=++cnt;
if(l==L&&r==R){/*sum[x]+=(r-l+1);*/ lzy[x]++; return;}//==
sum[x]+=(R-L+);
// pushdown(x,l,r);
int mid=((l+r)>>);
// if(mid>=L)add(ls[x],l,mid,L,R);//会重复计算sum!(因为标记永久化)
// if(mid<R)add(rs[x],mid+1,r,L,R);
// sum[x]=sum[ls[x]]+sum[rs[x]];
if(mid<L)add(rs[x],mid+,r,L,R);
else if(mid>=R)add(ls[x],l,mid,L,R);
else add(ls[x],l,mid,L,mid),add(rs[x],mid+,r,mid+,R);
}
ll ask(int x,int l,int r,int L,int R)
{
if(!x)return ;//
ll ret=lzy[x]*(R-L+);
if(l==L&&r==R)return ret+sum[x];//
// pushdown(x,l,r);
int mid=((l+r)>>);
// if(mid>=L)ret+=ask(x<<1,l,mid,L,R);
// if(mid<R)ret+=ask(x<<1|1,mid+1,r,L,R);//别把ls,rs和x<<1,x<<1|1写串了!
// if(mid>=L)ret+=ask(ls[x],l,mid);
// if(mid<R)ret+=ask(rs[x],mid+1,r); //会重复计算!!
// printf("ret=%lld\n",ret);
if(mid<L)return ret+ask(rs[x],mid+,r,L,R);
else if(mid>=R)return ret+ask(ls[x],l,mid,L,R);
else return ret+ask(ls[x],l,mid,L,mid)+ask(rs[x],mid+,r,mid+,R);
}
int query(int x,int l,int r,int k)
{
if(l==r)return l;
ll tmp=ask(rt[x<<|],,n,L,R),mid=((l+r)>>);//查询第k大
if(tmp>=k)return query(x<<|,mid+,r,k);//
else return query(x<<,l,mid,k-tmp);//
}
void insert(int x,int l,int r,int c)
{
add(rt[x],,n,L,R);
if(l==r)return;//
int mid=((l+r)>>);//权值区间
if(c<=mid)insert(x<<,l,mid,c);
else insert(x<<|,mid+,r,c);
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d%d%d",&tp[i],&a[i],&b[i],&c[i]);
if(tp[i]==)tmp[++tot]=c[i];
}
sort(tmp+,tmp+n+); tot=unique(tmp+,tmp+n+)-tmp-;
for(int i=;i<=m;i++)
{
L=a[i],R=b[i];
if(tp[i]==)
{
int tt=lower_bound(tmp+,tmp+tot+,c[i])-tmp;
insert(,,tot,tt);
}
else printf("%d\n",tmp[query(,,tot,c[i])]);
}
return ;
}
bzoj3110 [Zjoi2013]K大数查询——线段树套线段树的更多相关文章
- BZOJ3110[Zjoi2013]K大数查询(树状数组+整体二分)
3110 [Zjoi2013]K大数查询 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a ...
- bzoj3110: [Zjoi2013]K大数查询 【树套树,标记永久化】
//========================== 蒟蒻Macaulish:http://www.cnblogs.com/Macaulish/ 转载要声明! //=============== ...
- BZOJ3110 [Zjoi2013]K大数查询 树套树 线段树 整体二分 树状数组
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ3110 题意概括 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位 ...
- BZOJ3110[Zjoi2013]K大数查询——权值线段树套线段树
题目描述 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是 ...
- bzoj3110: [Zjoi2013]K大数查询 【cdq分治&树套树】
模板题,折腾了许久. cqd分治整体二分,感觉像是把询问分到答案上. #include <bits/stdc++.h> #define rep(i, a, b) for (int i = ...
- 【BZOJ3110】K大数查询(权值线段树套线段树+标记永久化,整体二分)
题意:有N个位置,M个操作.操作有两种,每次操作 如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是 ...
- [BZOJ3110] [Zjoi2013] K大数查询 (树套树)
Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置 ...
- BZOJ3110: [Zjoi2013]K大数查询
喜闻乐见的简单树套树= =第一维按权值建树状数组,第二维按下标建动态开点线段树,修改相当于第二维区间加,查询在树状数组上二分,比一般的线段树还短= =可惜并不能跑过整体二分= =另外bzoj上的数据有 ...
- 【树套树】bzoj3110 [Zjoi2013]K大数查询
题解很多,实现起来以外地简洁.内层的区间线段树上用了标记永久化. #include<cstdio> using namespace std; #define N 50001 struct ...
随机推荐
- Cuder - 用C++11封装的CUDA类
以前写cuda:初始化环境,申请显存,初始化显存,launch kernel,拷贝数据,释放显存.一个页面大部分都是这些繁杂但又必须的操作,有时还会忘掉释放部分显存. 今天用C++11封装了这些CUD ...
- zabbix配置邮件报警(第四篇)
zabbix配置邮件报警(第四篇) 邮件报警可采用本地邮件服务,也可以自定义脚本,这里我采用本地邮件服务报警 添加收件人
- 聊聊JS动画库:Velocity.js
前言 又到了炎热的7月,很久没有更新技术文章了,原因是上月月底实习结束,从公司离职.然后最近在弄自己的项目和考驾照,为了下次公司的应聘做准备,送别了女朋友到外地,哩哩啦啦半个月把一切事情都办妥后,还是 ...
- Java对象的创建及使用
Java对象的创建及使用 对象是类的具体实例(instance),是真实存在的个体:
- 考试T1总结(又CE?!)
考试T1CE... 最近不适合考试 T1 扶苏是个喜欢一边听古风歌一边写数学题的人,所以这道题其实是五三原题.歌曲中的主人公看着墙边的海棠花,想起当年他其实和自己沿着墙边种了一排海棠,但是如今都已枯萎 ...
- linq 升序排序 空值放后面并根据另一个字段进行多重排序
List<PickingInfo> res = GetDatas(); var _d = (from e in res select new { aa = e.pickingLibrary ...
- webpack-dev-middleware 与 webpack-hot-middlware
dev-middleware: live reload的实现: 思考一下我們要如何更新(live reload)呢? 當然是需要取得 webpack 編好的資料啊,於是就需要在從 request 到 ...
- 初学者对ASCII编码、Unicode编码、UTF-8编码的理解
最早的计算机在设计时采用8个比特(bit)作为一个字节(byte),所以,一个字节能表示的最大的整数就是 255(二进制 11111111=十进制 255),如果要表示更大的整数,就必须用更多的字节. ...
- jdk8--collect总结
https://blog.csdn.net/u014351782/article/details/53818430 一,collect是一个终端操作,它接收的参数是将流中的元素累积到汇总结果的各种方式 ...
- 00110_Class类
1.Class 对象是在加载类时由 Java虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的: 2.获取Class对象的三种方式 (1)方式一:通过Object类中的getObj ...