线段树维护区间前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 ...
随机推荐
- 五大好用的开源MySQL管理工具推荐
众所周知,对于数据库管理工作者(DBA)来说,保持数据正常运行在最佳状态需要具备敏捷.专注.冷静和快速的反应能力.因为数据库几乎是所有应用程序成功运行的核心,由于DBA负责组织数据,因此寻找可靠的工具 ...
- 平时碰到系统CPU飙高和频繁GC,你会怎么排查?
处理过线上问题的同学基本上都会遇到系统突然运行缓慢,CPU 100%,以及Full GC次数过多的问题.当然,这些问题的最终导致的直观现象就是系统运行缓慢,并且有大量的报警.本文主要针对系统运行缓慢这 ...
- 用例a失败,跳过测试用例b和c并标记失败xfail
前言 当用例a失败的时候,如果用例b和用例c都是依赖于第一个用例的结果,那可以直接跳过用例b和c的测试,直接给他标记失败xfail用到的场景,登录是第一个用例,登录之后的操作b是第二个用例,登录之后操 ...
- 【js】什么是函数节流与函数去抖
函数节流 意思:节省流量,不会一直访问. | 指定时间内不执行,指定时间后执行. | 一段时间内只执行一次 场景: 比如控制游戏人物攻击,时间内就算按得很快,也只能砍一刀,过后才能砍第二刀. 搜索引擎 ...
- centos7:storm集群环境搭建
1.安装storm 下载storm安装包 在线下载 wget http://apache.fayea.com/storm/apache-storm-1.1.1/apache-storm-1.1.1.t ...
- python+ selenium&APPium自动化 page Object 设计模式
题记: 之前公司项目比较稳定, 在进行了系统测试,想用自动化测试进行冒烟测试,或者对主要功能进行测试, 因此用到了PO模式 因此做个记录: Page Object Page Object模式是使用Se ...
- Redis的 SLAVEOF 命令
SLAVEOF host port SLAVEOF 命令用于在 Redis 运行时动态地修改复制(replication)功能的行为. 通过执行 SLAVEOF host port 命令,可以将当前服 ...
- 【监控笔记】【1.3】监控事件系列——SQL Trace(黑盒跟踪 BlackBox Trace)
[1]它跟踪了哪些事件? (1.1)存储过程执行(SP:Strating) (1.2)T-SQL执行(SQL:BatchString) (1.3)错误和警告(Exception,Attention) ...
- Widget代码讲解
参考:https://zhuanlan.zhihu.com/p/28225011 QT版本为5.12.4 1.main.cpp #include "widget.h" #inclu ...
- Intellij IDEA奇巧妙计(不停更新)
1,在pom.xml文件中,Ctrl+Shift+Alt+U打开Manven依赖视图 2,Alt+7 查看类里面方法,变量等结构 3, Shift+Esc 收缩编译提示框 4, ctrl+r 替换本页 ...