bzoj3110: [Zjoi2013]K大数查询 【树套树,标记永久化】
//==========================
蒟蒻Macaulish:http://www.cnblogs.com/Macaulish/ 转载要声明!
//==========================
好久没写题解了。
但是这题太神了然后做法太神了于是写一下。
这题做法很多,比如黄学长hzw的权值线段树套线段树,比如学长云的bit套主席树(其实是写法更神然后我不会用)。
然后看到hzhwcmhf大神题解。
http://tieba.baidu.com/p/2246783535
震惊了。
好了开说说做法。建一颗朴素的线段树,树的每个点表示每个区间,然后每个区间建两棵树,一棵是mark树,一棵是all树,两棵都是权值线段树。
“mark表示该区间每个点上都会加上mark线段树里的元素
all表示该区间所有点的元素集合”
这道题麻烦的地方就在与标记,如果标记要下传的话,那么每次可能要下传很多个点。
于是标记永久化,就是不要下传。
对于修改:如果修改区间是当前点区间,那么就加入mark中。如果不是,那么就加入all。操作都是单点修改。加入mark时为c+1,加入all是为c+len
对于询问:首先包含在查询区间内的区间的所有的all,然后mark要去掉没有重合在一起的地方,乘回差值。
(太久没写题解都不知道怎么说了)
速度还是挺快的
type
arr=record
all,mark:longint;
end;
const
maxn=;
var
tree:array[..maxn]of arr;
size,lson,rson,root1,root2:array[..maxn]of longint;
tot,sum,n,m,i,j,k,l:longint; function add:longint;
begin
inc(tot);
exit(tot);
end; procedure sinsert(var x:longint;y,z:longint);
var
left,right,mid,old:longint;
begin
if x= then x:=add;
old:=x;
left:=;
right:=n;
while left<=right do begin
mid:=(left+right)>>;
inc(size[x],z);
if left=right then break;
if y<=mid then begin
right:=mid;
if lson[x]= then lson[x]:=add;
x:=lson[x];
end
else begin
left:=mid+;
if rson[x]= then rson[x]:=add;
x:=rson[x];
end;
end;
x:=old;
end; procedure binsert(x,l,r,tl,tr,tc:longint);
var
mid:longint;
begin
if (tl=l) and (tr=r) then begin
sinsert(tree[x].mark,tc,);
exit;
end;
mid:=(l+r)>>;
sinsert(tree[x].all,tc,tr-tl+);
if tl>mid then binsert(x<<+,mid+,r,tl,tr,tc)
else
if tr<=mid then binsert(x<<,l,mid,tl,tr,tc)
else begin
binsert(x<<,l,mid,tl,mid,tc);
binsert(x<<+,mid+,r,mid+,tr,tc);
end;
end; procedure before(x,l,r,tl,tr:longint);
var
mid:longint;
begin
inc(sum);
root1[sum]:=tree[x].mark;
root2[sum]:=tr-tl+;
if (tl=l) and (tr=r) then begin
inc(sum);
root1[sum]:=tree[x].all;
root2[sum]:=;
exit;
end;
mid:=(l+r)>>;
if tl>mid then before(x<<+,mid+,r,tl,tr)
else
if tr<=mid then before(x<<,l,mid,tl,tr)
else begin
before(x<<,l,mid,tl,mid);
before(x<<+,mid+,r,mid+,tr);
end;
end; procedure query(x,y,z:longint);
var
left,right,now,mid:longint;
begin
sum:=;
before(,,n,x,y);
left:=;
right:=n;
while left<right do begin
now:=;
mid:=(left+right)>>;
for i:= to sum do
now:=size[rson[root1[i]]]*root2[i]+now;
if now<z then begin
dec(z,now);
right:=mid;
for i:= to sum do
root1[i]:=lson[root1[i]];
end
else begin
left:=mid+;
for i:= to sum do
root1[i]:=rson[root1[i]];
end;
end;
writeln(left);
end; begin
readln(n,m);
while m> do begin
dec(m);
readln(i,j,k,l);
if i= then binsert(,,n,j,k,l)
else query(j,k,l);
end;
readln;
readln;
end.
数组开小竟然wa了两次。
bzoj3110: [Zjoi2013]K大数查询 【树套树,标记永久化】的更多相关文章
- P3332 [ZJOI2013]K大数查询(线段树套线段树+标记永久化)
P3332 [ZJOI2013]K大数查询 权值线段树套区间线段树 把插入的值离散化一下开个线段树 蓝后每个节点开个线段树,维护一下每个数出现的区间和次数 为了防止MLE动态开点就好辣 重点是标记永久 ...
- BZOJ3110[Zjoi2013]K大数查询(树状数组+整体二分)
3110 [Zjoi2013]K大数查询 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a ...
- 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] [Zjoi2013] K大数查询 (树套树)
Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置 ...
- 【bzoj3110】[Zjoi2013]K大数查询 整体二分+树状数组区间修改
题目描述 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c.如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数 ...
- BZOJ3110: [Zjoi2013]K大数查询
喜闻乐见的简单树套树= =第一维按权值建树状数组,第二维按下标建动态开点线段树,修改相当于第二维区间加,查询在树状数组上二分,比一般的线段树还短= =可惜并不能跑过整体二分= =另外bzoj上的数据有 ...
- bzoj3110 [Zjoi2013]K大数查询——线段树套线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 外层权值线段树套内层区间线段树: 之所以外层权值内层区间,是因为区间线段树需要标记下传 ...
随机推荐
- LeetCode:40. Combination Sum II(Medium)
1. 原题链接 https://leetcode.com/problems/combination-sum-ii/description/ 2. 题目要求 给定一个整型数组candidates[ ]和 ...
- 利尔达CC3200模块第一篇之-wlan_ap例程测试
1. 本次采用利尔达的CC3200模块,CC3200主时钟80M,内部没有flash,必须外接SPI Flash.本次测试采用利尔达科技的CC3200的底板和模块(左边).烧写连接VCC, GND, ...
- Qt-LCD电子时钟
先上效果图吧 就是这个样子,简单的时间显示时间. 这里需要注意的是,我们最好建立一个空文件,这里我们需要建立一个集成QLCDNumber的类 具体方法如下图 一下是源代码 digiclock.h #i ...
- docker官网安装
最近发现一些同学居然不会安装docker,难,不难,只是“网络不好”! 如果是学习的话,目前我发现的比较好的方法还是到清华的开源镜像网站: https://mirror.tuna.tsinghua.e ...
- Java开发工程师(Web方向) - 01.Java Web开发入门 - 第2章.HTTP协议简介
第2章--HTTP协议简介 HTTP协议简介 Abstract: HTTP协议的特性,HTTP请求/响应的过程,HTTP请求/响应的报文格式等知识,最后会演示如何通过Chrome提供的开发者工具,去跟 ...
- 孤荷凌寒自学python第七十七天开始写Python的第一个爬虫7
孤荷凌寒自学python第七十七天开始写Python的第一个爬虫7 (完整学习过程屏幕记录视频地址在文末) 今天在上一天的基础上继续完成对我的第一个代码程序的书写. 今天的学习仍然是在纯粹对docx模 ...
- [Clr via C#读书笔记]Cp10属性
Cp10属性 属性的本质就是方法,只是看起来像字段罢了: 无参属性 就是一般属性: 字段一般要private,然后通过设置访问方法-访问器来访问:属性是方法语法变种:getset不一定要访问支持字段: ...
- OpenMPI源码剖析4:rte.h 头文件的说明信息
上一篇文章中说道,我们在 rte.h 中发现了有价值的说明: 我们一块一块来分析,首先看到第一块,关于 Process name Object: * (a) Process name objects ...
- 【QT】常用类
官方文档 doc QWidget QWidget类是所有用户界面对象的基类. 窗口部件是用户界面的一个基本单元:它从窗口系统接收鼠标.键盘和其它事件,并且在屏幕上绘制自己. 每一个窗口部件都是矩形的, ...
- python 读取 log日志的编码问题
1.我要读取log日志的”执行成功”的个数,log日志编码格式为GBK 2.显示报错,大致意思是说utf-8的代码不能解析log日志 3.后来想想把log日志用GBK编码读出来,写到新文件中,用utf ...