BZOJ_1798_[AHOI2009]维护序列_线段树

题意:老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成。 有长为N的数列,不妨设为a1,a2,…,aN 。有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一段数全部加一个值; (3)询问数列中的一段数的和,由于答案可能很大,你只需输出这个数模P的值。

分析:线段树上要打两个标记。要注意下传的顺序。显然先乘后加和先加后乘是不一样的。我们发现如果是先加后乘的话更改子树值的式子里会出现除法。不妨规定任何时候都先乘后加。推出的式子即为

t[lson]=(t[lson]*mul[pos]+add[pos]*(mid-l+1))%p;

mul[lson]=(mul[lson]*mul[pos])%p;

add[lson]=(add[lson]*mul[pos]+add[pos])%p;

代码:

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <algorithm>
  4. using namespace std;
  5. #define ls p<<1
  6. #define rs p<<1|1
  7. #define LL long long
  8. #define N 100050
  9. LL add[N<<2],t[N<<2],mul[N<<2],mod;
  10. int a[N],n,m;
  11. void bt(int l,int r,int p){
  12. mul[p]=1;
  13. if(l==r){
  14. scanf("%lld",&t[p]);return ;
  15. }
  16. int mid=l+r>>1;
  17. bt(l,mid,ls);bt(mid+1,r,rs);
  18. t[p]=(t[ls]+t[rs])%mod;
  19. }
  20. void pud(int l,int r,int p){
  21. int mid=l+r>>1;
  22. if(add[p]==0&&mul[p]==1)return ;
  23. t[ls]=(t[ls]*mul[p]+add[p]*(mid-l+1))%mod;
  24. t[rs]=(t[rs]*mul[p]+add[p]*(r-mid))%mod;
  25. mul[ls]=mul[ls]*mul[p]%mod;
  26. mul[rs]=mul[rs]*mul[p]%mod;
  27. add[ls]=(add[ls]*mul[p]+add[p])%mod;
  28. add[rs]=(add[rs]*mul[p]+add[p])%mod;
  29. mul[p]=1;add[p]=0;
  30. }
  31. void upad(int l,int r,int x,int y,int c,int p){
  32. if(x<=l&&y>=r){
  33. add[p]=(add[p]+c)%mod;
  34. t[p]+=1ll*(r-l+1)*c;t[p]%=mod;
  35. return ;
  36. }
  37. int mid=l+r>>1;
  38. pud(l,r,p);
  39. if(x<=mid)upad(l,mid,x,y,c,ls);
  40. if(y>mid)upad(mid+1,r,x,y,c,rs);
  41. t[p]=(t[ls]+t[rs])%mod;
  42. }
  43. void upmu(int l,int r,int x,int y,int c,int p){
  44. if(x<=l&&y>=r){
  45. mul[p]=mul[p]*c%mod;
  46. add[p]=add[p]*c%mod;
  47. t[p]=t[p]*c%mod;
  48. return ;
  49. }
  50. pud(l,r,p);
  51. int mid=l+r>>1;
  52. if(x<=mid)upmu(l,mid,x,y,c,ls);
  53. if(y>mid)upmu(mid+1,r,x,y,c,rs);
  54. t[p]=(t[ls]+t[rs])%mod;
  55. }
  56. LL query(int l,int r,int x,int y,int p){
  57. if(x<=l&&y>=r)return t[p];
  58. int mid=l+r>>1;
  59. LL re=0;
  60. pud(l,r,p);
  61. if(x<=mid)re=(re+query(l,mid,x,y,ls))%mod;
  62. if(y>mid)re=(re+query(mid+1,r,x,y,rs))%mod;
  63. return re;
  64. }
  65. int main(){
  66. scanf("%d%lld",&n,&mod);
  67. bt(1,n,1);
  68. scanf("%d",&m);
  69. int op,x,y,z;
  70. for(int i=1;i<=m;i++){
  71. scanf("%d",&op);
  72. if(op==1){
  73. scanf("%d%d%d",&x,&y,&z);
  74. upmu(1,n,x,y,z,1);
  75. }else if(op==2){
  76. scanf("%d%d%d",&x,&y,&z);
  77. upad(1,n,x,y,z,1);
  78. }else{
  79. scanf("%d%d",&x,&y);
  80. printf("%lld\n",query(1,n,x,y,1));
  81. }
  82. }
  83. }

BZOJ_1798_[AHOI2009]维护序列_线段树的更多相关文章

  1. 洛谷P2023 [AHOI2009]维护序列(线段树区间更新,区间查询)

    洛谷P2023 [AHOI2009]维护序列 区间修改 当我们要修改一个区间时,要保证 \(ax+b\) 的形式,即先乘后加的形式.当将区间乘以一个数 \(k\) 时,原来的区间和为 \(ax+b\) ...

  2. 1798. [AHOI2009]维护序列【线段树】

    Description 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2 ...

  3. 【题解】洛谷P2023 [AHOI2009] 维护序列(线段树)

    洛谷P2023:https://www.luogu.org/problemnew/show/P2023 思路 需要2个Lazy-Tag 一个表示加的 一个表示乘的 需要先计算乘法 再计算加法 来自你谷 ...

  4. BZOJ 1798: [Ahoi2009]Seq 维护序列seq( 线段树 )

    线段树.. 打个 mul , add 的标记就好了.. 这个速度好像还挺快的...( 相比我其他代码 = = ) 好像是#35.. ---------------------------------- ...

  5. bzoj 1798: [Ahoi2009]Seq 维护序列seq 线段树 区间乘法区间加法 区间求和

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeO ...

  6. Bzoj 1798: [Ahoi2009]Seq 维护序列seq(线段树区间操作)

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec Memory Limit: 64 MB Description 老师交给小可可一个维护数列的任务,现在小可 ...

  7. BZOJ1798: [Ahoi2009]Seq 维护序列seq[线段树]

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 5504  Solved: 1937[Submit ...

  8. bzoj 1798: [Ahoi2009]Seq 维护序列seq (线段树 ,多重标记下放)

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 7773  Solved: 2792[Submit ...

  9. 【BZOJ】1798: [Ahoi2009]Seq 维护序列seq 线段树多标记(区间加+区间乘)

    [题意]给定序列,支持区间加和区间乘,查询区间和取模.n<=10^5. [算法]线段树 [题解]线段树多重标记要考虑标记与标记之间的相互影响. 对于sum*b+a,+c直接加上即可. *c后就是 ...

随机推荐

  1. 解决 RabbitMQ 集群 Channel shutdown: connection error 错误(HAProxy 负载均衡)

    相关文章:搭建 RabbitMQ Server 高可用集群 具体错误信息: 2018-05-04 11:21:48.116 ERROR 60848 --- [.168.0.202:8001] o.s. ...

  2. Mongodb3.6 快速入门(一)

    一.安装 官网下载地址 文章中用的3.6community server 64 位. 安装完成后的目录如下: 二.启动Mongodb 有多种方式可以启动Mongodb,但在实际项目中,主要还是通过配置 ...

  3. JAVA程序员面试宝典

    程序员面试之葵花宝典 面向对象的特征有哪些方面    1. 抽象:抽象就是忽略一个主题中与当前目标2. 无关的那些方面,3. 以便更充分地注意与当前目标4. 有关的方面.抽象并不5. 打算了解全部问题 ...

  4. SQL Server复制表结构和表数据生成新表的语句

    参考:http://topic.csdn.net/t/20020621/09/820025.html SELECT   *   INTO   newTableName   FROM   oldTabl ...

  5. Ocelot中文文档-流量控制

    感谢@catcherwong 的文章激励我最终写出了这个文档 Ocelot支持上游的请求限制,以便您的下游服务不会过载. 此功能是由GitHub上的@geffzhang添加! 非常感谢. 好了,为了让 ...

  6. 第一章 python介绍、变量、数据类型、流程控制语句等

    一.python介绍 1.python的诞生 python是一种面向对象的解释型计算机程序设计语言,由荷兰人Guido van Rossum(龟叔)于1989年发明,第一个公开发行版发行于1991年. ...

  7. git 使用简易指南

  8. 第三方支付设计——账户体系

    第三方支付架构设计之-帐户体系 一,      什么是第三方支付?         什么是第三方支付?相信很多人对这个名字很熟悉,不管是从各种媒体等都经常听到,可以说是耳熟能熟.但,如果非得给这个名词 ...

  9. 图片与base64的互转

    /// <summary>        /// 把图片转换到文本信息        /// </summary>        /// <param name=&quo ...

  10. STL-Vector源码剖析

    G++ ,cygnus\cygwin-b20\include\g++\stl_vector.h 完整列表 /* * * Copyright (c) 1994 * Hewlett-Packard Com ...