本来只是想学一下CDQ,还是先把整体二分搞懂一点。


这题窝几个月前分别用划分树,树套树,主席树和挑战上介绍的分桶法实现了一发(然而现在都忘得差不多了)



最快的是划分树,其次是主席树,然后是树套树,还有一个10000+ms分桶法的思想实现的,最上面的就是整体二分的方法。


题目链接:

http://poj.org/problem?id=2104

题意:

给定序列,求每次询问区间的第K大数。

分析:

首先介绍整体二分。

看了这两个博客才明白一点。整体二分初步老顽童的代码

整体二分:

整体二分就是将所有询问一起二分,然后获得每个询问的答案。

整体二分的过程实质上是个按照数值来划分操作序列的过程。

对每一个询问我们都需要判定一下,以决定它被划分到哪一个答案的区间里。这个判定过程就是通过比较比二分的mid大的数的个数和k。

以下摘自许昊然论文-《浅谈数据结构题的几个非经典解法》

询问的答案具有二分性显然是前提。我们发现,因为修改判定标准的贡献相互独立,且贡献的值(如果有的话)与判定标准无关,所以如果我们已经计算过某一些修改对询问的贡献,那么这个贡献将永远不会改变,我们没有必要当判定标准改变时再次计算这些部分修改的贡献,只要记录下当前的总贡献,在进一步二分时,直接加上新的贡献即可。

这样处理的复杂度可以不再与序列总长度直接相关了,而可以只与当前处理的序列的长度相关

具体理解看代码。。。

时间复杂度:

定义T(C,S)表示待二分区间长度为C,待二分序列长度为S,不妨设单次处理时间复杂度O(f(n)),则有

T(C,S)=T(C/2,S0+T(C/2,S−S0))+O(f(s))

解之得

T(C,n)≤O(f(n)logC)

注:只有在O(f(n))≥O(n)时成立.

代码:

以POJ 2104为例

  1. #include<cstdio>
  2. #include<iostream>
  3. #include<cstring>
  4. using namespace std;
  5. const int maxn = 1e5 + 5, maxm = 1e4 + 5, oo = 0x3f3f3f3f;
  6. int n, m;
  7. int cnt;
  8. struct QUERY{
  9. int x, y, k;
  10. int id, type;
  11. };
  12. QUERY q[maxn + maxm], q1[maxn + maxm], q2[maxn + maxm];
  13. int bit[maxn];
  14. int ans[maxn];
  15. void add(int i, int x)
  16. {
  17. while(i <= n){
  18. bit[i] += x;
  19. i += i &-i;
  20. }
  21. }
  22. int sum(int i)
  23. {
  24. int res = 0;
  25. while(i){
  26. res += bit[i];
  27. i -= i & -i;
  28. }
  29. return res;
  30. }
  31. void init()
  32. {
  33. memset(bit, 0, sizeof(bit));
  34. cnt = 1;
  35. }
  36. void solve(int ql, int qr, int l, int r)
  37. {
  38. if(ql > qr) return;
  39. if(l == r){
  40. for(int i = ql; i <= qr; i++){
  41. if(q[i].type == 2) ans[q[i].id] = l;
  42. }
  43. return ;
  44. }
  45. int m = l + r >> 1;
  46. int p1 = 0, p2 = 0;
  47. for(int i = ql; i <= qr; i++){
  48. if(q[i].type == 1){
  49. if(q[i].x <= m){
  50. add(q[i].id, q[i].y);
  51. q1[p1++] = q[i];
  52. }else q2[p2++] = q[i];
  53. }else{
  54. int res = sum(q[i].y) - sum(q[i].x - 1);
  55. if(res >= q[i].k) q1[p1++] = q[i];
  56. else{
  57. q[i].k -= res;
  58. q2[p2++] = q[i];
  59. }
  60. }
  61. }
  62. //清空标记
  63. for(int i = 0; i < p1; i++){
  64. if(q1[i].type == 1) add(q1[i].id, -q1[i].y);
  65. }
  66. for(int i = 0; i < p1; i++){
  67. q[ql + i] = q1[i];
  68. }
  69. for(int i = 0; i < p2; i++){
  70. q[ql + p1 + i] = q2[i];
  71. }
  72. solve(ql, ql + p1 - 1, l , m);
  73. solve(ql + p1, qr, m + 1, r);
  74. }
  75. int main (void)
  76. {
  77. scanf("%d%d", &n, &m);
  78. init();
  79. int x, y, k;
  80. for(int i = 0; i < n; i++){
  81. scanf("%d", &x);
  82. q[cnt++] = (QUERY){x, 1, oo, i + 1, 1};
  83. }
  84. for(int i = 0; i < m; i++){
  85. scanf("%d%d%d", &x, &y, &k);
  86. q[cnt++] = (QUERY){x, y, k, i + 1, 2};
  87. }
  88. solve(1, cnt - 1, -oo, oo);
  89. for(int i = 1; i <= m; i++){
  90. printf("%d\n", ans[i]);
  91. }
  92. return 0;
  93. }

POJ 2104 K-th Number【整体二分 + 树状数组】的更多相关文章

  1. 【bzoj3110】[Zjoi2013]K大数查询 整体二分+树状数组区间修改

    题目描述 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c.如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数 ...

  2. BZOJ.3110.[ZJOI2013]K大数查询(整体二分 树状数组/线段树)

    题目链接 BZOJ 洛谷 整体二分求的是第K小(利用树状数组).求第K大可以转为求第\(n-K+1\)小,但是这样好像得求一个\(n\). 注意到所有数的绝对值\(\leq N\),将所有数的大小关系 ...

  3. 【BZOJ3110】【整体二分+树状数组区间修改/线段树】K大数查询

    Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个位置到第b个位 ...

  4. BZOJ_3110_[Zjoi2013]K大数查询_整体二分+树状数组

    BZOJ_3110_[Zjoi2013]K大数查询_整体二分+树状数组 Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位 ...

  5. 【BZOJ-2527】Meteors 整体二分 + 树状数组

    2527: [Poi2011]Meteors Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 831  Solved: 306[Submit][Stat ...

  6. 【bzoj4009】[HNOI2015]接水果 DFS序+树上倍增+整体二分+树状数组

    题目描述 给出一棵n个点的树,给定m条路径,每条路径有一个权值.q次询问求一个路径包含的所有给定路径中权值第k小的. 输入 第一行三个数 n和P 和Q,表示树的大小和盘子的个数和水果的个数. 接下来n ...

  7. 【bzoj2527】[Poi2011]Meteors 整体二分+树状数组

    题目描述 有N个成员国.现在它发现了一颗新的星球,这颗星球的轨道被分为M份(第M份和第1份相邻),第i份上有第Ai个国家的太空站. 这个星球经常会下陨石雨.BIU已经预测了接下来K场陨石雨的情况.BI ...

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

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

  9. BZOJ3110 [Zjoi2013]K大数查询 树套树 线段树 整体二分 树状数组

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ3110 题意概括 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位 ...

  10. 【POJ2104】【整体二分+树状数组】区间第k大

    Description You are working for Macrohard company in data structures department. After failing your ...

随机推荐

  1. 登录脚本重构Element

    登录脚本重构Element package com.gubai.selenium; import org.openqa.selenium.By; import org.openqa.selenium. ...

  2. HDU 5381 The sum of gcd (技巧,莫队算法)

    题意:有一个含n个元素的序列,接下来有q个询问区间,对每个询问区间输出其 f(L,R) 值. 思路: 天真单纯地以为是道超级水题,不管多少个询问,计算量顶多就是O(n2) ,就是暴力穷举每个区间,再直 ...

  3. Big Data Mindmap

  4. Python 使用re模块实现正则表达式

    # coding: utf-8 # Team : Quality Management Center # Author:Carson # Date :2019/6/21 10:41 # Tool :P ...

  5. 【整理】iview Tree数据格式问题,无限递归树处理数据

    iview Tree数据格式问题,无限递归树处理数据 https://juejin.im/post/5b51a8a4e51d455d6825be20

  6. 测试常用的linux命令

    一.系统 1.halt:         关机   poweroff: 关机 2.reboot:     重启 二.处理目录和文件的命令 1.ll:     显示文件详细信息 ls:    显示文件目 ...

  7. [BZOJ3940]:[Usaco2015 Feb]Censoring(AC自动机)

    题目传送门 题目描述: FJ把杂志上所有的文章摘抄了下来并把它变成了一个长度不超过105的字符串S.他有一个包含n个单词的列表,列表里的n个单词记为t1…tN.他希望从S中删除这些单词.FJ每次在S中 ...

  8. 前端拖动div 效果

    /** * author levi * url http://levi.cg.am */ $(function() { $(document).mousemove(function(e) { if(! ...

  9. vue 项目规范

    1, 组件化 2, css 分清单独和通用的 3, 封装请求 4, 命名原则 1: 尽量和后端保持一致 2: 简单常见的单词 3: 全部小写

  10. jquery ajax示例

    $.ajax({ type: "POST",//方法类型 dataType: "json",//预期服务器返回的数据类型 url: "${ctx }/ ...