介绍本题的两种做法:

方法1

前置芝士

  1. 线段树:一个很重要的数据结构.
  2. 树状数组:一个很重要的数据结构.

具体实现

区间修改,单点查询很容易就会想到树状数组了,至于查询前k个数的和又可以丢给权值线段树去干,所以第一种很显然的方法就是树状数组套一个线段树实现.

代码

  1. #include<bits/stdc++.h>
  2. #define REP(i,first,last) for(int i=first;i<=last;++i)
  3. #define DOW(i,first,last) for(int i=first;i>=last;--i)
  4. using namespace std;
  5. const int maxN=5e5+7;
  6. const int INF=2147483647;
  7. int N,M;
  8. int op[maxN];
  9. int s[maxN],e[maxN],p[maxN];
  10. int arr[maxN];
  11. int sor[maxN];
  12. int len=0;
  13. map<int,int>Hash;//数据很大需要离散化
  14. int val__[maxN];//记录离散化以后每个数代表的原来的值
  15. struct Tree//动态开点线段树
  16. {
  17. int sum,lson,rson;
  18. long long sum_;
  19. }tree[maxN*32];
  20. int point_cnt=0;
  21. //线段树标准define
  22. #define LSON tree[now].lson
  23. #define RSON tree[now].rson
  24. #define MIDDLE ((left+right)>>1)
  25. #define LEFT LSON,left,MIDDLE
  26. #define RIGHT RSON,MIDDLE+1,right
  27. void PushUp(int now)
  28. {
  29. tree[now].sum=tree[LSON].sum+tree[RSON].sum;//数的个数
  30. tree[now].sum_=tree[LSON].sum_+tree[RSON].sum_;//每个数相加以后的和
  31. }
  32. void UpDataMain(int num,int val,int &now,int left=1,int right=len)
  33. //修改,在now中加入val个num
  34. {
  35. if(num<left||right<num)
  36. {
  37. return;
  38. }
  39. if(!now)
  40. {
  41. now=++point_cnt;
  42. }
  43. if(left==right)
  44. {
  45. tree[now].sum+=+val;
  46. tree[now].sum_+=+val*val__[num];//需要加上原理的值*个数
  47. return;
  48. }
  49. UpDataMain(num,val,LEFT);
  50. UpDataMain(num,val,RIGHT);
  51. PushUp(now);
  52. }
  53. int lowbit(int now)//树状数组用的lowbit
  54. {
  55. return now&-now;
  56. }
  57. int root[maxN];
  58. void UpData(int top,int num,int val)//在top的位置加上val个num
  59. {
  60. for(int now=top;now<=N;now+=lowbit(now))//树状数组的修改
  61. {
  62. UpDataMain(num,val,root[now]);
  63. }
  64. }
  65. //记录下当前需要加上的树的当前节点
  66. int add_tree[maxN];
  67. int num_add=0;
  68. int GetSum()//当前树中数的个数
  69. {
  70. int sum=0;
  71. REP(i,1,num_add){sum+=tree[add_tree[i]].sum;}
  72. return sum;
  73. }
  74. long long GetSum_()//当前树的数的和
  75. {
  76. long long sum=0;
  77. REP(i,1,num_add){sum+=tree[add_tree[i]].sum_;}
  78. return sum;
  79. }
  80. int GetSumLeft()//当前树的左子树的数的个数
  81. {
  82. int sum=0;
  83. REP(i,1,num_add){sum+=tree[tree[add_tree[i]].lson].sum;}
  84. return sum;
  85. }
  86. long long GetSum_Left()//当前树的左子树的数的和
  87. {
  88. long long sum=0;
  89. REP(i,1,num_add){sum+=tree[tree[add_tree[i]].lson].sum_;}
  90. return sum;
  91. }
  92. int GetSumRight()//当前树的右子树的数的个数
  93. {
  94. int sum=0;
  95. REP(i,1,num_add){sum+=tree[tree[add_tree[i]].rson].sum;}
  96. return sum;
  97. }
  98. long long GetSum_Right()//当前树的右子树的数的和
  99. {
  100. long long sum=0;
  101. REP(i,1,num_add){sum+=tree[tree[add_tree[i]].rson].sum_;}
  102. return sum;
  103. }
  104. void GetRootLeft()//将节点换成左儿子
  105. {
  106. REP(i,1,num_add){add_tree[i]=tree[add_tree[i]].lson;}
  107. }
  108. void GetRootRight()//将节点换成右儿子
  109. {
  110. REP(i,1,num_add){add_tree[i]=tree[add_tree[i]].rson;}
  111. }
  112. long long QueryMain(int k,int left=1,int right=len)//查询部分主要函数
  113. {
  114. int sum=GetSum();//得到当前树的数的个数
  115. if(left==right)//如果是叶节点
  116. {
  117. return /*当前表示的数*/val__[left]*/*只有sum个数,最多取k个数,所以取一个min*/min(sum,k);
  118. }
  119. if(k>=sum)//如果k太大
  120. {
  121. return GetSum_();//返回当前树的数的和
  122. }
  123. int left_sum=GetSumLeft();//左子树的数的个数
  124. if(left_sum>=k)//如果大于等于k就差左子树
  125. {
  126. GetRootLeft();
  127. return QueryMain(k,left,MIDDLE);
  128. }
  129. //否则就差找右子树
  130. long long result=GetSum_Left();
  131. GetRootRight();
  132. return result+QueryMain(k-left_sum,MIDDLE+1,right);
  133. }
  134. void BeforeQuery(int place)//预处理
  135. {
  136. num_add=0;
  137. for(int now=place;now;now-=lowbit(now))
  138. {
  139. add_tree[++num_add]=root[now];
  140. }
  141. }
  142. long long Query(int place,int k)//查询
  143. {
  144. BeforeQuery(place);//预处理
  145. return QueryMain(k);
  146. }
  147. void UpDataAdd(int left,int right,int num)//修改,和普通树状数组相同
  148. {
  149. UpData(left,Hash[num],1);
  150. UpData(right+1,Hash[num],-1);
  151. }
  152. int main()
  153. {
  154. scanf("%d%d",&M,&N);
  155. int num_cnt=0;
  156. REP(i,1,M)
  157. {
  158. scanf("%d%d%d",&s[i],&e[i],&p[i]);//记录下来离散化
  159. sor[++num_cnt]=p[i];
  160. }
  161. sort(sor+1,sor+1+num_cnt);
  162. sor[0]=-INF;
  163. REP(i,1,num_cnt)
  164. {
  165. if(sor[i]!=sor[i-1])
  166. {
  167. Hash[sor[i]]=++len;
  168. val__[len]=sor[i];
  169. }
  170. }
  171. REP(i,1,M)
  172. {
  173. UpDataAdd(s[i],e[i],p[i]);//直接修改
  174. }
  175. long long pre=1;
  176. int x,a,b,c,k;
  177. REP(i,1,N)
  178. {
  179. scanf("%d%d%d%d",&x,&a,&b,&c);
  180. k=1+(a*pre+b)%c;//按公式计算k
  181. pre=Query(x,k);//查询
  182. printf("%lld\n",pre);
  183. }
  184. return 0;
  185. }

方法2

前置芝士

  1. 可持久化线段树 :可以用主席树来实现.
  2. 差分:优化方法1.

具体实现

可以发现方法1非常非常麻烦,所以可以发现可以用主席树维护前缀每个数出现的次数,然后就差分搞一下就好了.

代码

  1. #include<bits/stdc++.h>
  2. #define REP(i,first,last) for(int i=first;i<=last;++i)
  3. #define DOW(i,first,last) for(int i=first;i>=last;--i)
  4. using namespace std;
  5. const int maxN=3e5+7;
  6. int N,M;
  7. int sor[maxN];
  8. long long hanum[maxN];
  9. int p[maxN];
  10. int tot=0;
  11. map<int,int>Hash;
  12. //一个没什么用的东西,可以像搞图论一样把每个位置要放入的数和这个位置连一条边,可以简单处理
  13. struct Edge
  14. {
  15. int next,val,add;
  16. }edge[maxN*2];
  17. int cnt_edge=0;
  18. int head[maxN];
  19. #define FOR(now) for(int _i_=head[now];_i_;_i_=edge[_i_].next)
  20. #define VAL edge[_i_].val
  21. #define ADD edge[_i_].add
  22. void AddEdge(int form,int val,int add)
  23. {
  24. edge[++cnt_edge].val=val;
  25. edge[cnt_edge].add=add;
  26. edge[cnt_edge].next=head[form];
  27. head[form]=cnt_edge;
  28. }
  29. struct Tree//主席树
  30. {
  31. int lson,rson,sum;
  32. long long sum_;
  33. }tree[maxN*32];
  34. #define LSON tree[now].lson
  35. #define RSON tree[now].rson
  36. #define MIDDLE ((left+right)>>1)
  37. #define LEFT LSON,left,MIDDLE
  38. #define RIGHT RSON,MIDDLE+1,right
  39. #define NEW_LSON tree[new_tree].lson
  40. #define NEW_RSON tree[new_tree].rson
  41. int cnt_point=0;
  42. void PushUp(int now)
  43. {
  44. tree[now].sum=tree[LSON].sum+tree[RSON].sum;
  45. tree[now].sum_=tree[LSON].sum_+tree[RSON].sum_;
  46. }
  47. void UpData(int num,int val,int &new_tree,int now,int left=1,int right=tot)
  48. {
  49. if(num<left||right<num)
  50. {
  51. new_tree=now;
  52. return;
  53. }
  54. new_tree=++cnt_point;
  55. if(left==right)
  56. {//和方法一差不多
  57. tree[new_tree].sum=tree[now].sum+val;//加上数的个数
  58. tree[new_tree].sum_=tree[now].sum_+hanum[num]*val/*数的个数*这个数*/;
  59. return;
  60. }
  61. UpData(num,val,NEW_LSON,LEFT);
  62. UpData(num,val,NEW_RSON,RIGHT);
  63. PushUp(new_tree);
  64. }
  65. long long Query(int k,int now,int left=1,int right=tot)//查询写得比较随便
  66. {
  67. if(k<=0)return 0;//懒得分类讨论
  68. if(left==right)//到叶节点了直接计算
  69. {
  70. return min(k,tree[now].sum)/*取k和当期树中数的个数的小值*/*hanum[left];
  71. }
  72. if(k>=tree[now].sum)//如果k太大了
  73. {
  74. return tree[now].sum_;
  75. }
  76. return Query(k,LEFT)+Query(k-tree[LSON].sum,RIGHT);
  77. }
  78. int root[maxN];//记录每个位置的主席树的根节点
  79. int main()
  80. {
  81. scanf("%d%d",&M,&N);
  82. int s,e;
  83. REP(i,1,M)
  84. {
  85. scanf("%d%d%d",&s,&e,&p[i]);
  86. sor[i]=p[i];
  87. AddEdge(s,p[i],1);//添加边到这个数,和差分相同,l位置+1,r+1位置-1
  88. AddEdge(e+1,p[i],-1);
  89. }
  90. sort(sor+1,sor+1+M);//离散化
  91. sor[0]=114154;
  92. REP(i,1,M)
  93. {
  94. if(sor[i]!=sor[i-1])
  95. {
  96. Hash[sor[i]]=++tot;
  97. hanum[tot]=sor[i];
  98. }
  99. }
  100. int check;
  101. REP(i,1,N)//建树
  102. {
  103. if(head[i])//如果这个位置有加入新的数
  104. {
  105. check=1;//开始是从上一颗树变化,后来是自己修改自己
  106. FOR(i)//把数加入这个数
  107. {
  108. UpData(Hash[VAL],ADD,root[i],root[i-check]);
  109. check=0;
  110. }
  111. }
  112. else
  113. {
  114. root[i]=root[i-1];//没有就和上一个位置相同
  115. }
  116. }
  117. long long pre=1;
  118. int x,a,b,c,k;
  119. REP(i,1,N)
  120. {
  121. scanf("%d%d%d%d",&x,&a,&b,&c);
  122. k=1+(a*pre+b)%c;//计算k
  123. pre=Query(k,root[x]);
  124. printf("%lld\n",pre);
  125. }
  126. return 0;
  127. }

比较两种方法

方法1的做法更显然,很容易得出,但是\(N\log_2^2N\)的时间复杂度很容易TLE,方法二更容易写,但是需要用到差分,不一定可以直接想出来,跑起来比方法1快了很多.

「Luogu P3168 [CQOI2015]任务查询系统」的更多相关文章

  1. Luogu P3168 [CQOI2015]任务查询系统

    题目链接 \(Click\) \(Here\) 差分主席树,就是把主席树做成一个差分前缀和的形式,还是很容易想到的. 写主席树的时候几个注意点: 查询可能开始于所有任务之前,二分任务点要把左边界设置为 ...

  2. P3168 [CQOI2015]任务查询系统

    题目地址:P3168 [CQOI2015]任务查询系统 主席树的模板题 更模板的在这儿:P3834 [模板]可持久化线段树 1(主席树) 形象的说,P3834是"单点修改,区间查询" ...

  3. bzoj3932 / P3168 [CQOI2015]任务查询系统(主席树+差分)

    P3168 [CQOI2015]任务查询系统 看到第k小,就是主席树辣 对于每一段任务(a,b,k),在版本a的主席树+k,版本b+1的主席树-k 同一时间可能有多次修改,所以开个vector存操作, ...

  4. 洛谷 P3168 [CQOI2015]任务查询系统 解题报告

    P3168 [CQOI2015]任务查询系统 题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分. 超级计算机中的任务用三元组\((S_i,E_i,P_i) ...

  5. 洛谷P3168 [CQOI2015]任务查询系统 [主席树,差分]

    题目传送门 任务查询系统 题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任 ...

  6. ●洛谷P3168 [CQOI2015]任务查询系统

    题链: https://www.luogu.org/problemnew/show/P3168题解: 主席树 强制在线? 那就直接对每一个前缀时间建一个线段树(可持久化线段树),线段树维护优先度权值. ...

  7. P3168 [CQOI2015]任务查询系统(主席树)

    题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第Ei ...

  8. 洛谷P3168 [CQOI2015]任务查询系统

    #include<cstdio> #include<cstdlib> #include<algorithm> #include<cstring> #in ...

  9. p3168 [CQOI2015]任务查询系统(差分+主席树)

    恕我才学浅薄,一开始想到的是树状数组+线段树,然后看了题解才第一次见到了差分这种神奇的科技 仔细想想,主席树的本质不就是前缀和嘛,加上一个差分也是可以的,没想到真是罪过罪过 对时间维护一个差分 在Si ...

随机推荐

  1. 深入delphi编程理解之消息(一)WINDOWS原生窗口编写及消息处理过程

    通过以sdk方式编制windows窗口程序,对理解windows消息驱动机制和delphi消息编程有很大的帮助. sdk编制windows窗口程序的步骤: 1.对TWndClass对象进行赋值; 2. ...

  2. 分布式系统:CAP 理论的前世今生

    CAP 理论是分布式系统设计中的一个重要理论,虽然它为系统设计提供了非常有用的依据,但是也带来了很多误解.本文将从 CAP 诞生的背景说起,然后对理论进行解释,最后对 CAP 在当前背景下的一些新理解 ...

  3. hybird怎么实现的(核心webview)

    链接:https://blog.csdn.net/gongch0604/article/details/80510005

  4. java篇 之 集合

    集合   链接:https://blog.csdn.net/weixin_42504145/article/details/83119088 数组: java的数组既可以存储基本数据类型,也可以存储引 ...

  5. markdown列表

    Markdown 列表 Markdown 支持有序列表和无序列表. 无序列表使用星号(*).加号(+)或是减号(-)作为列表标记: * 第一项 * 第二项 * 第三项 + 第一项 + 第二项 + 第三 ...

  6. java 生成/解读 二维码

    package com.rails.util; import com.swetake.util.Qrcode; import jp.sourceforge.qrcode.QRCodeDecoder; ...

  7. 【C语言】将输入的10个数排序

    代码: #include <stdio.h> int main() { ], t; int i, j, max; printf("请输入10个数:\n"); ; i & ...

  8. Linux - 找到正在使用的 Shell 是哪个

    1. ps -p $$ 一个名为 "$$" (这是shell的特殊参数),表示当前你正在运行的 shell 实例的 PID 2. echo $0 3. echo $SHELL - ...

  9. 《Airbnb 早期BP》---创业学习--训练营直播第3课--HHR

    1,Airbnb:300亿美金. 一,BP 价值: 1,优秀的BP原则: (1)UCD原则:user centered design,用户为中心的设计.站在投资人视角,回答最关心的问题. (2)清晰原 ...

  10. cf 76 div2

    传送门 题意 t组样例 n个学生, x最多交换次数 a交换的第一个位置 b交换的第二个位置 最多是交换n-1次,不懂的话可以找个数列自己模拟一下: 然后其他的就是abs(a-b)+x;两个位置之间的距 ...