划分树(poj2104)

时间:2022-12-16 11:33:25

poj2104

题意:给出n个数,有m次查询,每次查询要你找出 l 到 r 中第 k 大的数;

思路:划分树模板题

划分树(poj2104)

上述图片展现了查询时如何往下递推的过程

其中ly表示 [sl,l) 中有多少个数进入了左子树,num[ceng][r]表示[sl,r]中有多少个数进入了左子树,total表示[l,r]中有多少个数进入了左子树。

代码:

#include<cstdio>
#include<algorithm>
using namespace std;
int s[][];
int st[];//排序后的数组
int num[][];//第i层前j个数有几个进入了左子树
void bt(int ceng,int l,int r){
if(l==r)//递归到l,r相等时
return;
int mid=(l+r)/;
int sum1=mid-l+;
for(int i=l;i<=r;i++){//计算该有多少个与中间值相等的数可以进入左子树
if(s[ceng][i]<st[mid])
sum1--;
}
int cnt1=,cnt2=;
for(int i=l;i<=r;i++){
if(i==l){
num[ceng][i]=;
}
else{
num[ceng][i]=num[ceng][i-];
} if(s[ceng][i]<st[mid]||s[ceng][i]==st[mid]&&sum1>){//如果当前数字小于中间数或者当前数等于中间数并且当前进入左子树并与中间数相等的数的数量小于限制数量时
int k1=l+cnt1++;
//printf("qqqq%d %d %d\n",ceng,cnt1,k1);
s[ceng+][k1]=s[ceng][i];
num[ceng][i]++;
if(s[ceng][i]==st[mid])//如果相等,则与中间数相等的数可以进入的位置又少了一个
sum1--;
}
else{//进入右子树
int k2=mid+cnt2++;
s[ceng+][k2]=s[ceng][i];
//printf("qqqq%d %d %d\n",ceng,cnt2,k2);
}
}
bt(ceng+,l,mid);//递归建树
bt(ceng+,mid+,r);
}
int query(int ceng,int sl,int sr,int l,int r,int k){
//printf("www%d %d %d\n",ceng,sl,sr);
if(sl==sr){//递归到叶子节点
//printf("qq%d %d %d\n",ceng,sl,s[ceng][sl]);
return s[ceng][sl];
}
int ly;
if(l==sl)
ly=;//ly代表该段的l前面有多少个数进入了左子树
else
ly=num[ceng][l-];
int total=num[ceng][r]-ly;//l到r之间有多少个数进入了左子树
if(total>=k){//该区间有大于k个数进入了左子树 ,则第k大的数一定在左子树里面
return query(ceng+,sl,(sl+sr)>>,sl+ly,sl+num[ceng][r]-,k);
//l=sl+ly;新的左范围等于边界sl+l前面的数进入左子树的个数
//r=sl+num[ceng][r]-1;新的右范围等于边界sl+前r个数中进入左子树的个数
} //为什么r!=sl+ly + k因为虽然连续,但不是有序的
else{
int lr=l-sl-ly+((sl+sr)>>)+;//新的左范围等于l前面的数的总数减去前面数进入左子树的个数加上右子数的开始位置
return query(ceng+,((sl+sr)>>)+,sr,lr,lr+r-l-total,k-total);
//新的右范围等于新的左范围加上l到r之间数的个数减去l和r之间的数进入左子树的个数
} }
int main(){
int n,m;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++){
scanf("%d",&s[][i]);
st[i]=s[][i];
}
sort(st+,st+n+);
bt(,,n);
while(m--){
int l,r,k;
scanf("%d%d%d",&l,&r,&k);
printf("%d\n",query(,,n,l,r,k));
}
return ;
}

划分树(poj2104)的更多相关文章

  1. 划分树 poj2104 hdu5249

    KPI Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submis ...

  2. poj2104&lpar;划分树模板&rpar;

    poj2104 题意 给出一个序列,每次查询一个区间,要求告诉这个区间排序后的第k个数. 分析 划分树模板,O(mlogn). 建树.根据排序之后的数组,对于一个区间,找到中点的数,将整个区间分为左右 ...

  3. poj2104(划分树模板)

    poj2104 题意 给出一个序列,每次查询一个区间,要求告诉这个区间排序后的第k个数. 分析 划分树模板,O(mlogn). 建树.根据排序之后的数组,对于一个区间,找到中点的数,将整个区间分为左右 ...

  4. poj2104&amp&semi;&amp&semi;poj2761 (主席树&amp&semi;&amp&semi;划分树)主席树静态区间第k大模板

    K-th Number Time Limit: 20000MS   Memory Limit: 65536K Total Submissions: 43315   Accepted: 14296 Ca ...

  5. 初学划分树,小见解之!POJ-2104&sol;HDU-2665

    划分树 本来是学主席树的,可怜我等巨弱观群巨博客难解fotle主席的思想精髓.于是学了一下划分树,嗯,花了一下午时间理解build(其实自己模拟一遍就通了),我很难理解为什么划分树会看不懂而能学会主席 ...

  6. hdu2665 &amp&semi;&amp&semi; poj2104划分树

    K-th Number Time Limit: 20000MS   Memory Limit: 65536K Total Submissions: 47066   Accepted: 15743 Ca ...

  7. poj2104 线段树 划分树

    学习:http://www.cnblogs.com/pony1993/archive/2012/07/17/2594544.html 划分树的build: 划分树是分层构建的,在构建的t层时,我们可以 ...

  8. POJ2104 k-th number 划分树

    又是不带修改的区间第k大,这次用的是一个不同的方法,划分树,划分树感觉上是模拟了快速排序的过程,依照pivot不断地往下划分,然后每一层多存一个toleft[i]数组,就可以知道在这一层里从0到i里有 ...

  9. poj2104 划分树 区间K大 在线 无修改

    博主sbit....对于高级数据结构深感无力,然后这些东西在OI竟然烂大街了,不搞就整个人都不好了呢. 于是我勇猛的跳进了这个大坑 ——sbit 区间K大的裸题,在线,无修改. 可以用归并树(\(O( ...

随机推荐

  1. HttpURLConnection类

    导语 java.net.HttpURLConnectin类是URLConnection类的抽象子类.它在处理协议为HTTP的URL时特别有效.具体而言,它通过它可以获取和设置请求方法,确定是否重定向, ...

  2. NOI 1&period;5 41&colon;数字统计

    描述 请统计某个给定范围[L, R]的所有整数中,数字2出现的次数. 比如给定范围[2, 22],数字2在数2中出现了1次,在数12中出现1次,在数20中出现1次,在数21中出现1次,在数22中出现2 ...

  3. 一次失败的Selenium chromedriver切换

    背景 Selenium webdriver一直使用Firefox作为浏览器来跑webtest, 但是最近发现ff有时会报超时的错误,于是想到使用chromedriver来提升稳定性.本想只把.fire ...

  4. Azure ARM &lpar;4&rpar; 开始创建ARM Resource Group并创建存储账户

    <Windows Azure Platform 系列文章目录> 好了,接下来我们开始创建Azure Resource Group. 1.我们先登录Azure New Portal,地址是: ...

  5. Java基础知识强化之网络编程笔记17:Android网络通信之 使用Http的Post方式读取网络数据(基于HTTP通信技术)

    使用Http的Post方式与网络交互通信.Post方式需要向网络传输一部分数据,同时具有输入流和输出流. 详见:Android(java)学习笔记210:采用post请求提交数据到服务器(qq登录案例 ...

  6. C&num; winForm资源文件实现多语言切换

    这是我目前看到过最简单的多语言切换了 操作步驟 介面上的多語 Step1.將表單的Localizable屬性設為True Step2.切換表單的Language屬性為欲使用的語系 設完後會在分頁標籤上 ...

  7. Keepalived概述和安装(1)

    一.Keepalived概述 本文主要了解开源高可用负载均衡集群利器keepalived,掌握keepalived的安装,运用keepalived配置高可用集群,并能够实现keepalived与负载均 ...

  8. ES5新增内容

    一.数组API实际开发中forEach和map会多一些=>是es6语法中的arrow function举例:(x) => x + 6相当于function(x){return x + 6; ...

  9. 第三篇、Python函数

    1.函数和过程的定义: 1) 函数定义:函数是逻辑结构化和过程化的一种编程方法. 2) 过程定义:过程就是简单特殊没有返回值的函数. 当一个函数/过程没有使用return显示的定义返回值时,pytho ...

  10. STM32 F4 DAC DMA Waveform Generator

    STM32 F4 DAC DMA Waveform Generator Goal: generating an arbitrary periodic waveform using a DAC with ...