POJ2104 K-th Number 静态区间第k最值 平方分割
干掉这道题的那一刻,我只想说:我终于**的AC了!!!
最终内存1344K,耗时10282ms,比起归并树、划分树以及其他各种黑科技,这个成绩并不算光彩⊙﹏⊙
但至少,从最初的无数次TLE到最终的AC,这过程见证了一个二分算法的艰辛优化
感谢国家,感谢XXTV,感谢《挑战程序设计竞赛》~( ̄▽ ̄)~*
先贴代码:
- ;
- ;
- ;
- ;
- const int maxV=1e9;
- int bucket[bktCount][bktSize];
- int unOrdered[bktSize*bktCount];
- int ordered[bktSize*bktCount];
- int N,K;
- #include <cstdio>
- #include <cstring>
- #include <algorithm>
- void init()
- {
- scanf("%d%d",&N,&K);
- memset(bucket[N>>bktDigit],0x7f,sizeof(bucket[N>>bktDigit]));
- ;i<N;i++)
- {
- scanf("%d",unOrdered+i);
- ordered[i]=unOrdered[i];
- bucket[i>>bktDigit][i&bktMaxIdx]=unOrdered[i];
- }
- using std::sort;
- int bktUsed=N>>bktDigit;
- sort(ordered,ordered+N);
- ;i<=bktUsed;i++) sort(bucket[i],bucket[i]+bktSize);
- }
- inline void enumerate(int _rangeL,int _rangeR,int _val,int& _notMore)
- {
- for(int i=_rangeL;i<=_rangeR;i++)
- if(unOrdered[i]<=_val) ++_notMore;
- }
- inline void countBucket(int _bktIdx,int _val,int& _notMore)
- {
- using std::upper_bound;
- int* ub=upper_bound(bucket[_bktIdx],bucket[_bktIdx]+bktSize,_val);
- _notMore+=(ub-bucket[_bktIdx]);
- }
- int ask(int _rangeL,int _rangeR,int _k) //k-th smallest
- {
- int digitL=_rangeL>>bktDigit;
- int digitR=_rangeR>>bktDigit;
- ;
- ;
- while(vL<vR)
- {
- ;
- ;
- if(digitL==digitR)
- enumerate(_rangeL,_rangeR,ordered[midV],notMore);
- else
- {
- ;i<digitR;i++)
- countBucket(i,ordered[midV],notMore);
- enumerate(_rangeL,((digitL+)<<bktDigit)-,ordered[midV],notMore);
- enumerate(digitR<<bktDigit,_rangeR,ordered[midV],notMore);
- }
- ;
- else vR=midV;
- }
- return ordered[vL];
- }
- int main()
- {
- init();
- ;i<K;i++)
- {
- int l,r,k;
- scanf("%d%d%d",&l,&r,&k);
- printf(,r-,k));
- }
- ;
- }
1、为什么统计notMore,而不是统计less或者两者都统计?
二分的过程中,缩减区间的关键是:
1、必须使可能成为最终解的值保留在二分区间中
2、每一次都必须使区间大小的确被缩减,以防陷入死循环
在这道题中,某个值x为解的条件是:less(x)<x && notMore(x)>=x
如果统计Less的话,上面的代码很难是保证第一条的
而如果两者都统计的话,表面上当x满足条件时即可跳出,可以减少二分所需的时间
但是事实上,这样做的代价就是统计的时间复杂度常数乘以2,总的来说得不偿失(会TLE)
2、二分的对象是什么?可否把maxValue和minValue作为二分的对象?
Answer:NO!!!
正确的做法是将原数组排好序,然后对这个有序数组二分
理由很简单:范围小。二分区间长不会超过1e5
如果对数值本身二分的话,minValue和maxValue最坏时分别会达到-1e9和+1e9,二分的时间代价是前者的1.9倍
3、平方分割必须是严格的么?
Answer:NO(*^__^*)(论如何用标点和颜文字表达语气( ̄. ̄))
设数据规模为N,每个桶的大小为B,则单次询问的时间复杂度为: O ( (N / B ) * log B + B )
当B = O ( ( N * log N ) ^ 0.5 ) 时,总的时间复杂度会比严格的平方分割小一些
代码中将B取为了1024正是为此。(顺便也方便了位运算)
B取512时效率会相对变差,B取256时干脆TLE
这道题更好的做法是归并树,比归并树还好的做法是划分树,不过这都是后话了,有时间慢慢填坑
POJ2104 K-th Number 静态区间第k最值 平方分割的更多相关文章
- POJ2104 K-th Number —— 静态区间第k小
题目链接:http://poj.org/problem?id=2104 K-th Number Time Limit: 20000MS Memory Limit: 65536K Total Sub ...
- poj2104&&poj2761 (主席树&&划分树)主席树静态区间第k大模板
K-th Number Time Limit: 20000MS Memory Limit: 65536K Total Submissions: 43315 Accepted: 14296 Ca ...
- HDU3473--Minimum Sum(静态区间第k大)
Minimum Sum Time Limit: 16000/8000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Tota ...
- 数据结构2 静态区间第K大/第K小
给定数组$A[1...N]$, 区间$[L,R]$中第$K$大/小的数的指将$A[L...R]$中的数从大到小/从小到大排序后的第$K$个. "静态"指的是不带修改. 这个问题有多 ...
- Dynamic Rankings || 动态/静态区间第k小(主席树)
JYF大佬说,一星期要写很多篇博客才会有人看 但是我做题没有那么快啊QwQ Part1 写在前面 区间第K小问题一直是主席树经典题=w=今天的重点是动态区间第K小问题.静态问题要求查询一个区间内的第k ...
- 主席树初步学习笔记(可持久化数组?静态区间第k大?)
我接触 OI也快1年了,然而只写了3篇博客...(而且还是从DP跳到了主席树),不知道我这个机房吊车尾什么时候才能摸到大佬们的脚后跟orz... 前言:主席树这个东西,可以说是一种非常畸形的数据结构( ...
- 静态区间第k大(归并树)
POJ 2104为例 思想: 利用归并排序的思想: 建树过程和归并排序类似,每个数列都是子树序列的合并与排序. 查询过程,如果所查询区间完全包含在当前区间中,则直接返回当前区间内小于所求数的元素个数, ...
- 主席树学习笔记(静态区间第k大)
题目背景 这是个非常经典的主席树入门题——静态区间第K小 数据已经过加强,请使用主席树.同时请注意常数优化 题目描述 如题,给定N个整数构成的序列,将对于指定的闭区间查询其区间内的第K小值. 输入输出 ...
- 主席树(静态区间第k大)
前言 如果要求一些数中的第k大值,怎么做? 可以先就这些数离散化,用线段树记录每个数字出现了多少次. ... 那么考虑用类似的方法来求静态区间第k大. 原理 假设现在要有一些数 我们可以对于每个数都建 ...
随机推荐
- Maven学习(3) - Maven和Eclipse集成和构建多模块Maven项目
最近在工作中越来越经常的用到了Maven作为项目管理和Jar包管理和构建的工具,感觉Maven的确是很好用的.而且要将Maven的功能最大发挥出来,多模块是一个很好的集成例子. 一个Maven项目包括 ...
- 最棒的Visual Studio扩展
isual Studio是微软公司推出的开发环境,Visual Studio可以用来创建Windows平台下的Windows应用程序和网络应用程序,也可以用来创建网络服务.智能设备应用程序和Offic ...
- leetcode distinct-subsequences(DP)
参考https://oj.leetcode.com/problems/distinct-subsequences 动态规划方程 dp[i][j]=dp[i-1][j-1]+dp[i-1][j] (s( ...
- 【转】SVN linux命令及 windows相关操作(二)
转自这里:http://www.uml.org.cn/pzgl/200904246.asp 1 安装及下载client 端 2 什么是SVN(Subversion)? 3 为甚么要用SVN? 4 怎么 ...
- iOS开发之获取WIFI信号强度
虽然各种直接获取信号强度的api都被封杀了.但是还有一个另类的黑魔法可以获取到.那就是遍历UIStatusBar了 - (void)getSignalStrength{ UIApplication * ...
- ArrStack——数组栈(procedure)
//数组栈,对于无法预料栈的长度情况下,可能会因为原分配数组不够长而导致数据溢出,或因为数组太长而浪费空间.但是操作快,不需要额外的操作.而链表与此想法,可以动态分配内存,但是要增加额外的操作. #i ...
- 匿名类型和Object转换
本文转载:http://www.cnblogs.com/JustRun1983/archive/2012/05/13/2497997.html net中的匿名类型非常好用, 但是开发中遇到一个问题,当 ...
- MySQL字符串函数
字符串大写和小写转换 MySQL 字符串大写和小写转化函数有两对: lower(), uppper() 和 lcase(), ucase() mysql> select lower('DDD') ...
- ssss
18:15 2013/6/18 未结束的事情1 应用API接口切换2 03in.com 项目管理授权 软件中不成功3 党校考核时间 段列出所有 10:51 2013/6/20 daili 应用修改1 ...
- 使用r.js进行前端repuirejs的合并压缩
安装 requirejs npm install -g requirejs 安装好后: 找到刚刚requirejs的安装目录,在该目录下找到r.js,并拷贝待压缩合并项目的根目录下 在项目根目录下创建 ...