题意:

一个长度为N的整数序列,编号0 - N - 1。进行Q次查询,查询编号i至j的所有数中,第K大的数是多少。

分析:

  仅仅就是一道整体二分的入门题而已,没听说过整体二分?

  其实就是一个分治的函数,但是呢,我所理解的,这是一个只分不治的过程。为什么?因为我们把数值域和操作域经过若干次划分划到最后,当数值域固定到一个值时,如果存在对应的操作域(此时应该是询问域)的结果就已经是这个值了,所以并不需要合并(治)。

  上面这段话可能被我复杂化了?那我们简单说。(以下是求区间第k小的步骤)

  这样的题一般都是给出一个序列,询问区间第k大/小。我们把这个序列看做添加一个数 a[i] 在位置 i 的操作。询问也是操作。

  然后就是一个递归函数,不断二分数值的范围(数值域),每次将属于 [ l,mid ] 这个范围的添加操作以原位置在树状数组中加1,直接划到左部(操作域),将 [ mid+1,r ]  直接划到右部(不碰树状数组),对于询问呢,假如在树状数组中查询到的,位于这个区间内的1的个数小于k,证明这个区间内,左部添加的数(不大于mid的)小于k个,所以这个询问的答案一定在右部添加的数中(也就是一定比mid要大),因此,假如这个区间内的数在左部有x个,我们需要在右部求出这个区间的第k-x小的数即可,即将次询问的k减去x,划分到右部,反之,若我们在树状数组上查出来的数大于等于k,证明答案会在左部被求出,直接划分到左部即可!

  我们无意间就维护了两个单调性,一个是加入顺序的单调性,另一个是数值的单调性,这二者的有机结合,恰好就是整体二分的优势,可以让其解决很多比较复杂,多处有二分性的题目。

  上面的解释可能大家看了都会很迷茫,但是研究代码会让你更明白。

代码:

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
const int inf=0x3f3f3f3f,MAXN=;
struct node{
int x,y,z,id,type;
}a[MAXN],al[MAXN],ar[MAXN];
int n,m,ans[MAXN];
struct bit{
int a[MAXN];
void add(int x,int y)
{for(;x<=n;x+=x&(-x)) a[x]+=y;}
int query(int x){
int ans=;for(;x;x-=x&(-x)) ans+=a[x];
return ans;}
}t;
void solve(int l,int r,int ql,int qr){
if(ql>qr||l>r) return ;//l和r是数值域
if(l==r){//ql,lr是操作域
for(int i=ql;i<=qr;i++)
ans[a[i].id]=l;return ;
} int mid=(l+r)>>,lal=,lar=;
for(int i=ql;i<=qr;i++)
if(a[i].type){
if(a[i].y<=mid){
t.add(a[i].x,a[i].z);
al[++lal]=a[i];
} else ar[++lar]=a[i];
} else{ int tmp;
tmp=t.query(a[i].y)-t.query(a[i].x-);
if(tmp>=a[i].z) al[++lal]=a[i];
else ar[++lar]=a[i],ar[lar].z-=tmp;
} for(int i=;i<=lal;i++)
if(al[i].type==) t.add(al[i].x,-al[i].z);
for(int i=,j=ql;i<=lal;i++,j++)
a[j]=al[i];
for(int i=,j=ql+lal;i<=lar;i++,j++)
a[j]=ar[i];solve(l,mid,ql,ql+lal-);
solve(mid+,r,ql+lal,qr);return ;
} int main(){
scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%d",&a[i].y),a[i].x=i,a[i].y=-a[i].y,
a[i].z=,a[i].type=;
scanf("%d",&m);
for(int i=,j=n+;i<=m;i++,j++)
scanf("%d%d%d",&a[j].x,&a[j].y,&a[j].z),a[j].x++,a[j].y++,
a[j].id=i,a[j].type=;
solve(-inf,inf,,n+m);
for(int i=;i<=m;i++)
printf("%d\n",-ans[i]); return ;
}

整体二分

  

51nod 1175 区间第k大 整体二分的更多相关文章

  1. 【XSY2720】区间第k小 整体二分 可持久化线段树

    题目描述 给你你个序列,每次求区间第\(k\)小的数. 本题中,如果一个数在询问区间中出现了超过\(w\)次,那么就把这个数视为\(n\). 强制在线. \(n\leq 100000,a_i<n ...

  2. POJ2104 K-th Number —— 区间第k小 整体二分

    题目链接:https://vjudge.net/problem/POJ-2104 K-th Number Time Limit: 20000MS   Memory Limit: 65536K Tota ...

  3. 静态区间第k小 - 整体二分

    蒟蒻终于学会整体二分啦! 思路 实现 丑陋无比的代码 #include <bits/stdc++.h> using namespace std; const int N = 200005; ...

  4. 【ZOJ2112】【整体二分+树状数组】带修改区间第k大

    The Company Dynamic Rankings has developed a new kind of computer that is no longer satisfied with t ...

  5. ZOJ 1112 Dynamic Rankings【动态区间第K大,整体二分】

    题目链接: http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1112 题意: 求动态区间第K大. 分析: 把修改操作看成删除与增加 ...

  6. Dynamic Rankings——带修改区间第k大

    三种做法:1.整体二分: 二分mid 考虑小于mid的修改的影响 但是大于mid的修改可能会干掉小于mid的一些值 所以额外把一个修改变成一个值的删除和一个值的添加 这样就相互独立了! 整体二分,树状 ...

  7. 解决区间第K大的问题的各种方法

    例题:http://poj.org/problem?id=2104 最近可能是念念不忘,必有回响吧,总是看到区间第k大的问题,第一次看到是在知乎上有人面试被弄懵了后来又多次在比赛中看到.以前大概是知道 ...

  8. Permutation UVA - 11525(值域树状数组,树状数组区间第k大(离线),log方,log)(值域线段树第k大)

    Permutation UVA - 11525 看康托展开 题目给出的式子(n=s[1]*(k-1)!+s[2]*(k-2)!+...+s[k]*0!)非常像逆康托展开(将n个数的所有排列按字典序排序 ...

  9. 静态区间第k大(归并树)

    POJ 2104为例 思想: 利用归并排序的思想: 建树过程和归并排序类似,每个数列都是子树序列的合并与排序. 查询过程,如果所查询区间完全包含在当前区间中,则直接返回当前区间内小于所求数的元素个数, ...

随机推荐

  1. 关于JAVA通过REST接口对arcGis Server数据进行增删改查

    一: 添加要素 public void create(BoxVo boxVo) throws Exception { // 创建HTTP客户端 CloseableHttpClient httpclie ...

  2. Elasticsearch 2.3.2 安装部署

    先按照http://blog.csdn.net/love13135816/article/details/51690280这个教程安装, 不过后面的IK分词器安装部分有问题. 所以中文分词器插件的安装 ...

  3. typedef struct和struct 的区别 用途

    刚刚想到的,我们在用结构体的时候会遇到'->'和'.',这是什么情况呢? 不能混用的(c和c++不同语言对它们没有影响)   我说的不能混用的意思是'.'用于结构体指针的指向......而'-& ...

  4. mqtt遇到的问题锦集

    1.无效客户机标识 (2) Connect指令中的KeepAlive有效范围[60秒,300秒],否则会拒绝连接. 2.消息回调出现频繁的断开连接 待解决 3.长时间消息回调出现 已断开连接 (321 ...

  5. _bzoj2243 [SDOI2011]染色【树链剖分】

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=2243 裸的树链剖分,最开始我保存一个线段树节点的color值时(若有多种颜色则为-1),不小 ...

  6. vijos1846 [NOIP2013] 华容道【最短路】

    传送门:https://vijos.org/p/1983 (其实noip的题各个oj都会有的,就不贴其它传送门了) 这道题真的是,怎么说,我都不知道怎么评价了= =.果然数据量小的题怎么暴力都可以过. ...

  7. 题解报告:hdu 2191 悼念512汶川大地震遇难同胞——珍惜现在,感恩生活(多重背包)

    Problem Description 急!灾区的食物依然短缺!为了挽救灾区同胞的生命,心系灾区同胞的你准备自己采购一些粮食支援灾区,现在假设你一共有资金n元,而市场有m种大米,每种大米都是袋装产品, ...

  8. PWA之serviceWorker应用

    1.serviceWorker介绍service worker是一段运行在浏览器后台的JavaScript脚本,在页面中注册并安装成功后,它可以拦截和处理网络请求,实现缓存资源并可在离线时响应用户的请 ...

  9. Css 基本的规则写法

    样式表的写法: css的语法由一些标志构成,就是一个基本的样式表由选择器,属性和属性值构成.Css有标准的写法规则标准的css写法: h1 { Font-family:黑体;} h1:表示选择符Fon ...

  10. Thinkphp3.23 连接MSSQL方法

    Thinkphp 3.23要连接MSSQL,必须配置下,以下是主要的步骤. 1.要安装Microsoft Drivers for PHP for SQL Server驱动 下载驱动以前,要查看一下ph ...