线段树维护区间前k小
线段树维护区间前k小
$ solution: $
觉得超级钢琴太麻烦?在这里线段树提供一条龙服务 。
咳咳,开始讲正题!这道题我们有一个和超级钢琴复杂度一样 $ O(\sum x\times logn)~ $ 的做法。因为线段数支持动态维护最小值,而取 $ max $ 操作我们可以用线段树的 $ lazytag $ 实现(不懂可以看看代码里的标记下传和区间修改)。所以我们主要目的就是输出区间前 $ x $ 小,这个其实我们可以用线段树的单点修改完成!
我们在区间 $ [l,r] $ 里面找最小值,假设其下标为 $ k $ 我们记录它的位置和它的权值。然后我们暂时将这个位置在线段树上用单点修改操作改成 $ inf $ ,这样我们就不会再将这个位置作为最小值。然后我们再在 $ [l,r] $ 中找一个最小值,这时我们可以找到另一个位置 $ k_2 $ ,然后重复给它变成 $ inf $ 的操作,并记录它的位置和权值。这样不断循环,当我们找完所有的区间前 $ x $ 小后。我们再用线段树的单点修改操作,将线段树上对应位置的值改回来!然后这题就做完了!
注意每一个最小值我们都要花 $ log $ 的时间查找,更改为 $ inf $ ,最后再改回来。 复杂度中有一定常数,但是算法只用了线段树,没有其它数据结构鱼龙混杂,跑起来效果还不错,当然重点是码量小一些!
$ code: $
#include<iostream>
#include<cstdio>
#include<iomanip>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<ctime>
#include<cmath>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define ll long long
#define db double
#define rg register int
#define zuo k<<1,l,mid
#define you k<<1|1,mid+1,r
#define midd int mid=(l+r)>>1
#define pushd push(k,k<<1,k<<1|1)
using namespace std;
int n,m;
int x,y,v,t,sx,sv;
int as[500005];
inline int qr(){
register char ch; register bool sign=0; rg res=0;
while(!isdigit(ch=getchar()))if(ch=='-')sign=1;
while(isdigit(ch))res=res*10+(ch^48),ch=getchar();
if(sign)return -res; else return res;
}
struct su{
int da,id;
inline void min(const su a){
if(a.da<da){da=a.da;id=a.id;}
}
}a[500005];
struct tree{
int da[500005<<4];
int lz[500005<<4];
inline void build(int k,int l,int r){ //建树
if(l==r){da[k]=qr(); return ;}
midd; build(zuo); build(you);
da[k]=min(da[k<<1],da[k<<1|1]);
}
inline void push(int k,int l,int r){ //下传标记
if(lz[k]>da[l]){
da[l]=max(da[l],lz[k]);
lz[l]=max(lz[l],lz[k]);
}
if(lz[k]>da[r]){
da[r]=max(da[r],lz[k]);
lz[r]=max(lz[r],lz[k]);
} lz[k]=0;
}
inline void add1(int k,int l,int r){ //区间取max
if(x<=l&&r<=y){
if(da[k]>=v)return ; //权值只增不降
da[k]=max(da[k],v);
lz[k]=max(lz[k],v);
return ;
}if(lz[k])pushd; midd;
if(x<=mid)add1(zuo);
if(y>mid) add1(you);
da[k]=min(da[k<<1],da[k<<1|1]);
}
inline void add2(int k,int l,int r){ //单点修改权值
if(l==r){da[k]=sv; return ;}
if(lz[k])pushd; midd;
if(sx<=mid) add2(zuo);
else add2(you);
da[k]=min(da[k<<1],da[k<<1|1]);
}
inline su ask(int k,int l,int r){ //询问区间里的最小值信息
if(l==r)return su{da[k],l};
if(lz[k])pushd; midd;
if(x<=l&&r<=y){ //注意我们要精确的找到最小值位置
if(da[k<<1]==da[k])return ask(zuo);
else return ask(you);
}
register su res; res.da=1e9; res.id=1001;
if(x<=mid)res=ask(zuo);
if(y>mid) res.min(ask(you));
return res;
}
}tr;
int main(){
n=qr(); tr.build(1,1,n); m=qr();
for(rg i=1;i<=m;++i){
rg f=qr(); x=qr(); y=qr(); v=qr();
if(f==1){ //区间修改
tr.add1(1,1,n);
} else{
t=qr(); //
for(rg j=1;j<=t;++j){
a[j]=tr.ask(1,1,n); //读取最小值位置及权值
if(a[j].da>=v){t=j;break;} //不符合题意
sx=a[j].id; sv=1e9; //让最小值消失
tr.add2(1,1,n); //让之前的最小值不再被选中
}
if(a[t].da>=v) printf("-1");
else for(rg j=1;j<=t;++j) printf("%d ",a[j].da);
for(rg j=1;j<=t;++j){
sx=a[j].id; sv=a[j].da; //将之前改的变回原值
tr.add2(1,1,n);
}puts("");
}
}
return 0;
}
线段树维护区间前k小的更多相关文章
- 主席树--动态区间第k小
主席树--动态区间第\(k\)小 模板题在这里洛谷2617. 先对几个问题做一个总结: 阅读本文需要有主席树的基础,也就是通过区间kth的模板题. 静态整体kth: sort一下找第k小,时间复杂度\ ...
- [csu/coj 1080]划分树求区间前k大数和
题意:从某个区间内最多选择k个数,使得和最大 思路:首先题目给定的数有负数,如果区间前k大出现负数,那么负数不选和更大,于是对于所有最优选择,负数不会出现,所以用0取代负数,问题便转化为区间的前k大数 ...
- POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 )
POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 ) 题意分析 给出n个点,m个询问,和当前位置pos. 先给出n-1条边,u->v以及边权w. 然后有m个询问 ...
- A - 低阶入门膜法 - K-th Number (主席树查询区间第k小)
题目链接:https://cn.vjudge.net/contest/284294#problem/A 题目大意:主席树查询区间第k小. 具体思路:主席树入门. AC代码: #include<i ...
- HDU 5919 - Sequence II (2016CCPC长春) 主席树 (区间第K小+区间不同值个数)
HDU 5919 题意: 动态处理一个序列的区间问题,对于一个给定序列,每次输入区间的左端点和右端点,输出这个区间中:每个数字第一次出现的位子留下, 输出这些位子中最中间的那个,就是(len+1)/2 ...
- Can you answer these queries V SPOJ - GSS5 (分类讨论+线段树维护区间最大子段和)
recursion有一个整数序列a[n].现在recursion有m次询问,每次她想知道Max { A[i]+A[i+1]+...+A[j] ; x1 <= i <= y1 , x2 &l ...
- CodeForces - 587E[线段树+线性基+差分] ->(线段树维护区间合并线性基)
题意:给你一个数组,有两种操作,一种区间xor一个值,一个是查询区间xor的结果的种类数 做法一:对于一个给定的区间,我们可以通过求解线性基的方式求出结果的种类数,而现在只不过将其放在线树上维护区间线 ...
- hdu_5726_GCD(线段树维护区间+预处理)
题目链接:hdu_5726_GCD 题意: 给你n个数(n<=1e5)然后m个询问(m<=1e5),每个询问一个区间,问你这个区间的GCD是多少,并且输出从1到n有多少个区间的GCD和这个 ...
- 滑动窗口(poj,线段树维护区间最值)
题目描述 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. 例如: The array i ...
随机推荐
- 方法一破解:Excel工作表保护密码
在excel2016中实测验证过有效 在Excel中,为了保护自已的工作表不被修改,我们可以添加保护密码. 操作步骤: 1.把Excel文件的扩展名xlsx修改为Rar.瞬间Excel文件变成了压缩包 ...
- OpenCV学习笔记(1)
一.读入图像 使用cv2.imread()读入图像,图像应该在此程序的工作路径,第二个参数是告诉函数应该如何读取这幅图片 cv2.IMREAD_COLOR:读入一副彩色图像.图像的透明度会被忽略,这是 ...
- 九、SpringBoot集成Thymeleaf模板引擎
Thymeleaf咋读!??? 呵呵,是不是一脸懵逼...哥用我的大学四级英文知识告诉你吧:[θaimlif]. 啥玩意?不会音标?...那你就这样叫它吧:“赛母李府”,大部分中国人是听不出破绽的.. ...
- 关于加快INSERT语句执行速度和HINT /*+ append */及/*+ append nologging */的使用
(非归档模式下)创建表T01: SQL> create table t01 as select * from dba_objects where 1=2; Table created. (非归档 ...
- 使用pycharm编写python乱码
开始总是乱码,该设置的都设置了,后来用charde检测编码也一直报错,之后重启了pycharm就好了,乱码问题也没了
- Java -- 泛型父类中获取子类的泛型T
原文:https://blog.csdn.net/u014723529/article/details/70574026 /** * 获取实体类型名称 * 子类可覆盖此方法,返回:泛型T的类名.cla ...
- 2018 icpc 徐州
A 矩阵树定理可以用于最小生成树计数,最直观的做法就是求个mst,再用矩阵树定理求最小生成树个数,但是n<=1e5,显然不是o(n^3)可以做出来的. 考虑随机数据生成器,固定1e5的边,但是边 ...
- LeetCode.874-走路机器人模拟(Walking Robot Simulation)
这是悦乐书的第335次更新,第360篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第205题(顺位题号是874).网格上的机器人从点(0,0)开始并朝北.机器人可以接收三 ...
- CF 686D. Kay and Snowflake
给你一个树N个点,再给出Q个询问,问以x为根的子树中,重心是哪个?2≤n≤300000,1≤q≤30000 Sol:从下到上,根据性质做一下.1:如果某个点x,其子树y的大小超过总结点个数一半,则重心 ...
- 【DSP开发】【Linux开发】基于ARM+DSP进行应用开发
针对当前应用的复杂性,SOC芯片更好能能满足应用和媒体的需求,集成众多接口,用ARM做为应用处理器进行多样化的应用开发和用户界面和接口,利用DSP进行算法加速,特别是媒体的编解码算法加速,既能够保持算 ...