POJ 2104 K-th Number ( 求取区间 K 大值 || 主席树 || 离线线段树)
题意 : 给出一个含有 N 个数的序列,然后有 M 次问询,每次问询包含 ( L, R, K ) 要求你给出 L 到 R 这个区间的第 K 大是几
分析 :
求取区间 K 大值是个经典的问题,可以使用的方法有很多,我听过的只有主席树、整体二分法、划分树、分块……
因为是看《挑战》书介绍的平方分割方法(分块),所以先把分块说了,其他的坑以后再填
分块算法思想是将区间分为若干块,一般分为 n1/2 块然后在每块维护所需信息,可以把复杂度降到 O(根号n)
具体的分析和代码在《挑战程序设计竞赛》有很详细的解释,这里说一下代码的实现细节
题目在实现的时候用的是这种 [L, R) 左闭右开区间,这样的区间表示法在 STL 和 JAVA的类库中很常用
这样有很多优点,其中一个优点就是区间的长度是L ~ R,而判断两个区间的交或者并的时候思考的难度也降低很多。
L < R代表区间有值,L == R代表区间到了最后。用闭区间就特别麻烦,下面我给出的代码就是用闭区间的,纠结了我好久...
#include<vector> #include<stdio.h> #include<algorithm> using namespace std; ; ; vector<int> bucket[maxn / B]; int num[maxn], arr[maxn]; int N, M; int main(void) { while(~scanf("%d %d", &N, &M)){ ; i<N; i++){ scanf("%d", &arr[i]); bucket[i / B].push_back(arr[i]); num[i] = arr[i]; } sort(num, num + N); ; i<N/B; i++) sort(bucket[i].begin(), bucket[i].end()); int L, R, K; while(M--){ scanf("%d %d %d", &L, &R, &K); L--, R--; , ub = N - , ans = -; while(ub >= lb){ ); ; int TL = L, TR = R; > TL && TL % B != ) if(arr[TL++] <= num[mid]) c++; > TL && (TR+) % B != ) if(arr[TR--] <= num[mid]) c++; while(TR >= TL){ c += upper_bound(bucket[TL/B].begin(), bucket[TL/B].end(), num[mid]) - bucket[TL/B].begin(); TL += B; } ; ; } printf("%d\n", num[ans]); } } ; }
2018-05-07 更新
省赛被 可持久化Trie 打爆,决定学习一下可持久化数据结构
学了主席树,离线求取 K 大值,注意一下离散化
#include<stdio.h> #include<algorithm> using namespace std; ; ]; int root[maxn], sz; void Insert(int pre, int cur, int p, int l, int r) { if(l == r){ Node[cur].v = Node[pre].v + ; return ; } ); if(p <= m){ Node[cur].lc = ++sz; Node[cur].rc = Node[pre].rc; Insert(Node[pre].lc, Node[cur].lc, p, l, m); }else{ Node[cur].rc = ++sz; Node[cur].lc = Node[pre].lc; Insert(Node[pre].rc, Node[cur].rc, p, m+, r); } Node[cur].v = Node[Node[cur].lc].v + Node[Node[cur].rc].v; } int query(int L, int R, int l, int r, int k) { if(l == r) return l; ); int tmp = Node[Node[R].lc].v - Node[Node[L].lc].v; if(tmp >= k) return query(Node[L].lc, Node[R].lc, l, m, k); else , r, k-tmp); } int arr[maxn]; int mp[maxn]; int main(void) { int N, M; scanf("%d %d", &N, &M); ; i<N; i++) scanf("%d", &arr[i]), mp[i] = arr[i]; sort(mp, mp+N); int len = unique(mp, mp+N) - mp; ; i<=N; i++){ ]) - mp; root[i] = ++sz; Insert(root[i-], root[i], x+, , len); } int i, j, k; while(M--){ scanf("%d %d %d", &i, &j, &k); ], root[j], , len, k); printf(]); } ; }
POJ 2104 K-th Number ( 求取区间 K 大值 || 主席树 || 离线线段树)的更多相关文章
- [luoguP1440] 求m区间内的最小值(单调队列 || 线段树)
传送门 这种水题没必要搞线段树了,单调队列就行啊. ——代码 #include <cstdio> ; , t = ; int a[MAXN], q[MAXN]; int main() { ...
- 线性时间求取第 K 大数
求 Top K 的算法主要有基于快速排序的和基于堆的这两种,它们的时间复杂度都为 \(O(nlogK)\).借助于分治思想,以及快速排序的区间划分,我们可以做到 \(O(n)\) 时间复杂度.具体算法 ...
- poj 1523Tarjan算法的含义——求取割点可以分出的连通分量的个数
poj 1523Tarjan算法的含义——求取割点可以分出的连通分量的个数 题目大意:如题目所示 给你一些关系图——连通图,想要问你有没有个节点,损坏后,可以生成几个互相独立的网络(也就是连通分量), ...
- hdu6003 Problem Buyer 贪心 给定n个区间,以及m个数,求从n个区间中任意选k个区间,满足m个数都能在k个区间中找到一个包含它的区间,如果一个区间包含了x,那么 该区间不能再去包含另一个数,即k>=m。求最小的k。如果不存在这样的k,输出“IMPOSSIBLE!”。
/** 题目:hdu6003 Problem Buyer 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6003 题意:给定n个区间,以及m个数,求从n个区 ...
- 求余区间的求和类问题 离线+线段树 HDU4228
题目大意:给一个数组a,他的顺序是严格的单调增,然后有如下三个操作 ①加入一个val到a数组里面去,加入的位置就是a[i-1]<val<a[i+1] ②删除一个a[i]=val的值 ③查询 ...
- Splay(区间翻转)&树套树(Splay+线段树,90分)
study from: https://tiger0132.blog.luogu.org/slay-notes P3369 [模板]普通平衡树 #include <cstdio> #inc ...
- HDU 5700 区间交 离线线段树
区间交 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5700 Description 小A有一个含有n个非负整数的数列与m个区间.每个区间可以表示为 ...
- [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)
[BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...
- bzoj 3110 [Zjoi2013]K大数查询——线段树套线段树(标记永久化)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 第一道线段树套线段树! 第一道标记永久化! 为什么为什么写了两个半小时啊…… 本想线段 ...
随机推荐
- eslint 修改规则 函数形参不使用报错
函数再定义形参以后未使用就会出现报错的问题,需要设置一项eslint 规则 再.eslintrc.js rules里面添加 "no-unused-vars": "of ...
- 【ABAP系列】SAP 系统的消息类型分析 MESSAGE TYPE
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[ABAP系列]SAP 系统的消息类型分析 ME ...
- Vue-3D-Model:用简单的方式来展示三维模型
为什么做这个组件 我经常听到前端朋友们抱怨,在网页上展示三维模型太麻烦了.但是这方面的需求又有很多,例如做房地产的需要展示户型.卖汽车的需要展示汽车模型等. 在网页上展示三维模型就只能用WebGL技术 ...
- ant buid.xml 模板
<?xml version="1.0" encoding="UTF-8"?> <project name="ant" de ...
- SVN的各种符号含义,svn的星号,感叹号,问号等含义
黄色感叹号(有冲突):--这是有冲突了,冲突就是说你对某个文件进行了修改,别人也对这个文件进行了修改,别人抢在你提交之前先提交了,这时你再提交就会被提示发生冲突,而不允许你提交,防止你的提交覆盖了别人 ...
- MySQL-快速入门(10)触发器
1.什么是触发器 触发器是与表有关的命名数据库对象.触发器由事件来触发某个操作. 触发器是特殊的存储过程,触发器不需要call语句来调用,也不需要手工启动,只需要确定一个预定义的事件发生的时候,就会被 ...
- Labview 错误1400-打包库封装类时将对类重命名导致
现象 今天遇到了一个神奇的BUG,主程序调用了一个包含类的打包库,打包库中将字符串还原为类句柄时报错. 调用程序结构如下:. 主程序中将类句柄转化为XML字符串程序如下: 打包库内将字符串还原为句柄程 ...
- 4G 内存怎么读取一个 5G 的数据?
方法一:可以通过生成器,分多次读取,每次读取数量相对少的数据(比如 500MB)进行处理,处理结束后在读取后面的 500MB 的数据. 方法二:可以通过 linux 命令 split 切割成小文件,然 ...
- docker elk
1.核心组成 ELK由Elasticsearch.Logstash和Kibana三部分组件组成: Elasticsearch是个开源分布式搜索引擎,它的特点有:分布式,零配置,自动发现,索引自动分片, ...
- jdk8中几个核心的函数式接口笔记
1. Function接口 /** * function 接口测试 * function 函数只能接受一个参数,要接受两个参数,得使用BiFunction接口 */ public class Func ...