Description

一个可重复数字集合S的神秘数定义为最小的不能被S的子集的和表示的正整数。例如S={1,1,1,4,13},

1 = 1

2 = 1+1

3 = 1+1+1

4 = 4

5 = 4+1

6 = 4+1+1

7 = 4+1+1+1

8无法表示为集合S的子集的和,故集合S的神秘数为8。

现给定n个正整数a[1]..a[n],m个询问,每次询问给定一个区间l,r,求由a[l],a[l+1],…,a[r]所构成的可重复数字集合的神秘数。

Input

第一行一个整数n,表示数字个数。

第二行n个整数,从1编号。

第三行一个整数m,表示询问个数。

以下m行,每行一对整数l,r,表示一个询问。

Output

对于每个询问,输出一行对应的答案。

Sample Input

5

1 2 4 9 10

5

1 1

1 2

1 3

1 4

1 5

Sample Output

2

4

8

8

8

HINT

对于100%的数据点,n,m <= 100000,∑a[i] <= 10^9


思路

这是一个不套路的主席树题...真的是好题

首先发现一个性质,如果当前可以凑出来的区间是\([1,x]\)

那么如果有一个y需要被加进集合中

如果\(y\le x + 1\),那么新的可以表示出来的区间一定是\([1,x+y]\)

否则的话\(x+1\)一定不能被表示出来

但是这样需要按大小顺序考虑每一个数

所以就可以用主席数可持久化一下

然后每次因为在\([1,x+1]\)之间的数都可以被计入贡献

所以可以直接求一个前缀sum就可以了

每次就比较一下,如果不能更新答案或者没有合法答案就可以退出了

复杂度很迷我不会证明


  1. //Author: dream_maker
  2. #include<bits/stdc++.h>
  3. using namespace std;
  4. //----------------------------------------------
  5. //typename
  6. typedef long long ll;
  7. //convenient for
  8. #define fu(a, b, c) for (int a = b; a <= c; ++a)
  9. #define fd(a, b, c) for (int a = b; a >= c; --a)
  10. #define fv(a, b) for (int a = 0; a < (signed)b.size(); ++a)
  11. //inf of different typename
  12. const int INF_of_int = 1e9;
  13. const ll INF_of_ll = 1e18;
  14. //fast read and write
  15. template <typename T>
  16. void Read(T &x) {
  17. bool w = 1;x = 0;
  18. char c = getchar();
  19. while (!isdigit(c) && c != '-') c = getchar();
  20. if (c == '-') w = 0, c = getchar();
  21. while (isdigit(c)) {
  22. x = (x<<1) + (x<<3) + c -'0';
  23. c = getchar();
  24. }
  25. if (!w) x = -x;
  26. }
  27. template <typename T>
  28. void Write(T x) {
  29. if (x < 0) {
  30. putchar('-');
  31. x = -x;
  32. }
  33. if (x > 9) Write(x / 10);
  34. putchar(x % 10 + '0');
  35. }
  36. //----------------------------------------------
  37. const int N = 1e5 + 10;
  38. const int M = 2e6 + 10;
  39. int rt[N], ls[M], rs[M], tot = 0;
  40. ll sum[M];
  41. void build(int &t, int l, int r) {
  42. t = ++tot;
  43. ls[t] = rs[t] = sum[t] = 0;
  44. if (l == r) return;
  45. int mid = (l + r) >> 1;
  46. build(ls[t], l, mid);
  47. build(rs[t], mid + 1, r);
  48. }
  49. void insert(int &t, int last, int l, int r, int pos, int vl) {
  50. t = ++tot;
  51. ls[t] = ls[last];
  52. rs[t] = rs[last];
  53. sum[t] = sum[last] + vl;
  54. if (l == r) return;
  55. int mid = (l + r) >> 1;
  56. if (pos <= mid) insert(ls[t], ls[last], l, mid, pos, vl);
  57. else insert(rs[t], rs[last], mid + 1, r, pos, vl);
  58. }
  59. ll query(int t, int last, int l, int r, int pos) {
  60. if (l == r) return sum[t] - sum[last];
  61. int mid = (l + r) >> 1;
  62. if (pos <= mid) return query(ls[t], ls[last], l, mid, pos);
  63. else return sum[ls[t]] - sum[ls[last]] + query(rs[t], rs[last], mid + 1, r, pos);
  64. }
  65. int n, m, q, a[N], b[N], pre[N];
  66. int find_pow(int vl) {
  67. int l = 1, r = m, res = 0;
  68. while (l <= r) {
  69. int mid = (l + r) >> 1;
  70. if (pre[mid] <= vl) res = mid, l = mid + 1;
  71. else r = mid - 1;
  72. }
  73. return res;
  74. }
  75. int main() {
  76. //freopen("input.txt", "r", stdin);
  77. Read(n);
  78. fu(i, 1, n) {
  79. Read(a[i]);
  80. pre[i] = a[i];
  81. }
  82. sort(pre + 1, pre + n + 1);
  83. m = unique(pre + 1, pre + n + 1) - pre - 1;
  84. pre[m + 1] = INF_of_int;
  85. build(rt[0], 1, m);
  86. fu(i, 1, n) {
  87. b[i] = lower_bound(pre + 1, pre + m + 1, a[i]) - pre;
  88. insert(rt[i], rt[i - 1], 1, m, b[i], a[i]);
  89. }
  90. Read(q);
  91. while (q--) {
  92. int l, r; Read(l), Read(r);
  93. ll now = 0;
  94. while (1) {
  95. int pos = find_pow(now + 1);
  96. if (!pos) {++now; break;}
  97. int s = query(rt[r], rt[l - 1], 1, m, pos);
  98. if (s <= now) {
  99. ++now;
  100. break;
  101. }
  102. now = s;
  103. }
  104. Write(now), putchar('\n');
  105. }
  106. return 0;
  107. }

BZOJ4408: [Fjoi 2016]神秘数【主席树好题】的更多相关文章

  1. BZOJ 4408: [Fjoi 2016]神秘数 主席树 + 神题

    Code: #include<bits/stdc++.h> #define lson ls[x] #define mid ((l+r)>>1) #define rson rs[ ...

  2. BZOJ4408&4299[Fjoi 2016]神秘数——主席树

    题目描述 一个可重复数字集合S的神秘数定义为最小的不能被S的子集的和表示的正整数.例如S={1,1,1,4,13},1 = 1 2 = 1+1 3 = 1+1+1 4 = 4 5 = 4+1 6 = ...

  3. 【bzoj4408】[Fjoi 2016]神秘数 主席树

    题目描述 一个可重复数字集合S的神秘数定义为最小的不能被S的子集的和表示的正整数.例如S={1,1,1,4,13},1 = 12 = 1+13 = 1+1+14 = 45 = 4+16 = 4+1+1 ...

  4. BZOJ 4408: [Fjoi 2016]神秘数 [主席树]

    传送门 题意: 一个可重复数字集合S的神秘数定义为最小的不能被S的子集的和表示的正整数.例如S={1,1,1,4,13},8无法表示为集合S的子集的和,故集合S的神秘数为8.现给定n个正整数a[1]. ...

  5. [BZOJ4408][Fjoi 2016]神秘数

    [BZOJ4408][Fjoi 2016]神秘数 试题描述 一个可重复数字集合S的神秘数定义为最小的不能被S的子集的和表示的正整数.例如S={1,1,1,4,13},1 = 12 = 1+13 = 1 ...

  6. 【BZOJ4408】[Fjoi 2016]神秘数 主席树神题

    [BZOJ4408][Fjoi 2016]神秘数 Description 一个可重复数字集合S的神秘数定义为最小的不能被S的子集的和表示的正整数.例如S={1,1,1,4,13},1 = 12 = 1 ...

  7. BZOJ4408 [Fjoi 2016]神秘数 【主席树】

    题目链接 BZOJ4408 题解 假如我们已经求出一个集合所能凑出连续数的最大区间\([1,max]\),那么此时答案为\(max + 1\) 那么我们此时加入一个数\(x\),假若\(x > ...

  8. bzoj4408 [Fjoi 2016]神秘数 & bzoj4299 Codechef FRBSUM 主席树+二分+贪心

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4299 https://lydsy.com/JudgeOnline/problem.php?id ...

  9. Bzoj 4408: [Fjoi 2016]神秘数 可持久化线段树,神题

    4408: [Fjoi 2016]神秘数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 177  Solved: 128[Submit][Status ...

随机推荐

  1. MyBtis—原理及初始化

    Mybatis的功能架构分为三层: 1)       API接口层:提供给外部使用的接口API,开发人员通过这些本地API来操纵数据库.接口层 一接收到调用请求就会调用数据处理层来完成具体的数据处理. ...

  2. 2018-2019 ACM-ICPC, Asia Xuzhou Regional Contest Solution

    A. Rikka with Minimum Spanning Trees 题意: 给出一个图,求最小生成树的个数和权值 思路: 因为数据随机,只有一个MST #include <bits/std ...

  3. Uva11374 Dijkstra

    机场快线是市民从市内去机场的首选交通工具.机场快线分为经济线和商业线两种,线路.速度和价格都不同,你有一张商业线车票,可以坐一站商业线,而其他时候,只能乘坐经济线.假设换乘时间忽略不计,你的任务是找一 ...

  4. MySQL从删库到跑路(一)——MySQL数据库简介

    作者:天山老妖S 链接:http://blog.51cto.com/9291927 一.MySQL简介 1.MySQL简介 MySQL是一个轻量级关系型数据库管理系统,由瑞典MySQL AB公司开发, ...

  5. SQLServer中char、varchar、nchar、nvarchar比较

    转自:http://www.cnblogs.com/bluesky_blog/archive/2009/07/31/1535722.html 对于程序中的string型字段,SQLServer中有ch ...

  6. FM/AM收音机原理

    收音机这东西很早就开始用了,但一直都没有了解过它的原理,听说是很简单.下面记录一些笔记. 1. 基本概念 收音机是一种小型的无线电接收机,主要用于接受无线电广播节目,收听无线电发射台.首先说一下收音机 ...

  7. Centos75 firewalld防火墙

    Centos75 防火墙iptables被firewalld取代 #启动firewalld systemctl start firewalld #查看firewalld systemctl statu ...

  8. Windows自带计算器快捷键

    今天乱翻的时候发现了这个东西,下面就是各个快捷键: (以下功能在计算器面板上均能找到) 按键 功能 F9 \(-/+\) R 1/x @ \(\sqrt{}\) Ctrl+Shift+D 清除历史记录 ...

  9. C# .NET 开发心得

    1. 工作路径问题 1. 多项目构成的解决方案,Web APP作为启动项目时的工作路径 //当前执行的exe文件名 //C:\\Program Files\\IIS Express\\iisexpre ...

  10. POJ 1780 Code(欧拉回路+非递归dfs)

    http://poj.org/problem?id=1780 题意:有个保险箱子是n位数字编码,当正确输入最后一位编码后就会打开(即输入任意多的数字只有最后n位数字有效)……要选择一个好的数字序列,最 ...