题目

一个比较自然的想法是线段树维护二进制分组。

因为我们询问的是一段连续的操作的积,所以我们可以建一棵线段树,每个节点存储当前区间各个操作的积。

这里的操作的积指的是把一系列操作做完之后区间每个位置的变换。因为有很多连续的变换是一样的所以我们可以把它们缩起来。

因为我们知道\(k\)个操作最多会把整个区间划为\(2k+1\)段,所以所有节点的区间的总数是\(O(n\log n)\)级别的。

而合并两个线段树节点的操作可以使用归并排序。

然后我们查询就可以找到对应的\(log\ n\)的线段树上的节点,然后在每个节点存储的区间中二分找到查询的位置所在的区间,然后把所有查询到的变换乘起来就行了。

还要注意一件事情:

因为是强制在线,我们的操作也是一个个给出而不是一开始全部给出,所以我们需要一开始先建出空的线段树,然后新加进来一个操作,我们就把它插入到线段树上对应的\(log\ n\)个节点中去,同时pushup一下。

  1. #include<bits/stdc++.h>
  2. #define ep emplace_back
  3. #define pi pair<int,int>
  4. #define fi first
  5. #define se second
  6. #define ls p<<1
  7. #define rs p<<1|1
  8. using namespace std;
  9. namespace IO
  10. {
  11. char ibuf[(1<<21)+1],obuf[(1<<21)+1],st[15],*iS,*iT,*oS=obuf,*oT=obuf+(1<<21);
  12. char Get(){return (iS==iT? (iT=(iS=ibuf)+fread(ibuf,1,(1<<21)+1,stdin),(iS==iT? EOF:*iS++)):*iS++);}
  13. void Flush(){fwrite(obuf,1,oS-obuf,stdout),oS=obuf;}
  14. void Put(char x){*oS++=x;if(oS==oT)Flush();}
  15. int read(){int x=0,c=Get();while(!isdigit(c))c=Get();while(isdigit(c))x=x*10+c-48,c=Get();return x;}
  16. void write(int x){int top=0;if(!x)Put('0');while(x)st[++top]=(x%10)+48,x/=10;while(top)Put(st[top--]);Put('\n');}
  17. }
  18. using namespace IO;
  19. int max(int a,int b){return a>b? a:b;}
  20. const int N=100007;
  21. int n,m,q,ison,v[N],L[N<<2],R[N<<2];pi e(1,0),f;struct node{int l,r;pi x;node(int a=0,int b=0,pi c=e){l=a,r=b,x=c;}};
  22. vector<int>id[N];vector<node>t[N<<2],tmp;
  23. int operator<(node a,node b){return a.r<b.r;}
  24. pi mul(pi a,pi b){return pi(1ll*a.fi*b.fi%m,(1ll*a.se*b.fi+b.se)%m);}
  25. vector<node>merge(vector<node>a,vector<node>b)
  26. {
  27. vector<node>c;
  28. for(int i=0,j=0;i<(int)a.size()||j<(int)b.size();)
  29. if(a[i].r==b[j].r) c.ep(max(a[i].l,b[j].l),a[i].r,mul(a[i].x,b[j].x)),++i,++j;
  30. else if(a[i].r<b[j].r) c.ep(max(a[i].l,b[j].l),a[i].r,mul(a[i].x,b[j].x)),++i;
  31. else c.ep(max(a[i].l,b[j].l),b[j].r,mul(a[i].x,b[j].x)),++j;
  32. return c;
  33. }
  34. void build(int p,int l,int r)
  35. {
  36. id[r].ep(p),L[p]=l,R[p]=r,t[p].ep(1,n,e);
  37. if(l==r) return ;int mid=(l+r)>>1;
  38. build(ls,l,mid),build(rs,mid+1,r);
  39. }
  40. pi query(int p,int l,int r,int x)
  41. {
  42. if(l<=L[p]&&R[p]<=r) return lower_bound(t[p].begin(),t[p].end(),(node){0,x,e})->x;
  43. int mid=(L[p]+R[p])>>1;
  44. return mul((l<=mid? query(ls,l,r,x):e),(r>mid? query(rs,l,r,x):e));
  45. }
  46. int main()
  47. {
  48. ison=read()&1,n=read(),m=read();
  49. for(int i=1;i<=n;++i) v[i]=read();
  50. q=read(),build(1,1,min(q,100000));
  51. for(int i=1;i<=min(q,100000);++i) reverse(id[i].begin(),id[i].end());
  52. for(int T=0,l,r,a,b,ans=0;q;--q)
  53. {
  54. if(read()==1)
  55. {
  56. l=read(),r=read(),a=read(),b=read(),tmp.clear(),++T;
  57. if(ison) l^=ans,r^=ans;
  58. if(l>1) tmp.ep(1,l-1,e);
  59. tmp.ep(l,r,pi(a,b));
  60. if(r<n) tmp.ep(r+1,n,e);
  61. for(int p:id[T]) t[p]=L[p]==R[p]? merge(t[p],tmp):merge(t[ls],t[rs]);
  62. }
  63. else
  64. {
  65. l=read(),r=read(),a=read();
  66. if(ison) l^=ans,r^=ans,a^=ans;
  67. f=query(1,l,r,a),write(ans=(1ll*v[a]*f.fi+f.se)%m);
  68. }
  69. }
  70. return Flush(),0;
  71. }

UOJ46 玄学的更多相关文章

  1. uoj46玄学

    复杂度辣鸡没人权 疯狂爆oj 感觉要被众多uoj用户骂了 #include <bits/stdc++.h> #define ll long long #define LS ls[now]? ...

  2. 【BZOJ3821/UOJ46】玄学(二进制分组,线段树)

    [BZOJ3821/UOJ46]玄学(二进制分组,线段树) 题面 BZOJ UOJ 题解 呜,很好的题目啊QwQ. 离线做法大概可以线段树分治,或者直接点记录左右两次操作时的结果,两个除一下就可以直接 ...

  3. [UOJ46][清华集训2014]玄学

    uoj description 给出\(n\)个变换,第\(i\)个变换是将区间中\(l_i,r_i\)的数\(x\)变成\((a_ix+b_i)\mod m\). 每次会新增一个变换,或者查询询问如 ...

  4. UOJ46. 【清华集训2014】玄学

    传送门 Sol 考虑对于操作时间建立线段树,二进制分组 那么现在主要的问题就是怎么合并信息 你发现一个性质,就是每个修改只会在整个区间内增加两个端点 那么我们二进制分组可以得到每个区间内最多只有区间长 ...

  5. UOJ46. 【清华集训2014】玄学 [线段树,二进制分组]

    UOJ 思路 模拟赛出了这题,结果我没学过二进制分组--一波主席树然后空间就爆炸了-- 用线段树维护时间序列,每个节点维护\(a_i\to x_i\times a_i+b_i,i\in [1,n]\) ...

  6. UOJ46 清华集训2014玄学(线段树)

    注意到操作有结合律,容易想到用一个矩形表示第i次操作对第j个位置的数的影响.那么修改是单行内的区间修改,而查询是单列内的区间查询.这样二维线段树上以列为外层行为内层直接打标记就可以维护.然后就喜闻乐见 ...

  7. UOJ46 【清华集训2014】玄学 【时间线段树】

    题目链接:UOJ 这题的时间线段树非常的妙. 对时间建立线段树,修改的时候在后面加,每当填满一个节点之后就合并进它的父亲. 对于一个节点维护序列,发现这是一个分段函数,合并就是归并排序.于是就形成了差 ...

  8. 洛谷P2832 行路难 分析+题解代码【玄学最短路】

    洛谷P2832 行路难 分析+题解代码[玄学最短路] 题目背景: 小X来到了山区,领略山林之乐.在他乐以忘忧之时,他突然发现,开学迫在眉睫 题目描述: 山区有n座山.山之间有m条羊肠小道,每条连接两座 ...

  9. C++玄学预编译优化

    #pragma GCC diagnostic error "-std=c++11" #pragma GCC optimize("-fdelete-null-pointer ...

随机推荐

  1. numpy中np.linalg.norm()求向量、矩阵的范数

    np.linalg.norm() # linalg = linear(线性) + algebra(代数),   norm表示范数 x_norm = np.linalg.norm(x, ord=None ...

  2. Another Filling the Grid

    E. Another Filling the Grid 参考:Codeforces Round #589 (Div. 2)-E. Another Filling the Grid-容斥定理 容斥这个东 ...

  3. 记一次sparkOnyarn错误:java.lang.UnsatisfiedLinkError

    错误大概这样: Caused by: java.util.concurrent.ExecutionException: Boxed Error Caused by: java.lang.Unsatis ...

  4. JavaWeb-SpringSecurity初认识

    Spring Security 安全 百度百科 功能:Spring Security对Web安全性的支持大量地依赖于Servlet过滤器.这些过滤器拦截进入请求,并且在应用程序处理该请求之前进行某些安 ...

  5. Linux安装JDK、tomcat

    修改tomcat 相关配置必须重启后才生效 如何启动tomcat 在终端框内切换到tomcat 的bin路径下 启动tomcat:./startup.sh 关闭tomcat:./shutdown.sh ...

  6. Java多线程深入理解

    在java中要想实现多线程,有两种手段,一种是继续Thread类,另外一种是实现Runable接口. 对于直接继承Thread的类来说,代码大致框架是: ? 1 2 3 4 5 6 7 8 9 10 ...

  7. (转载)Zookeeper的功能以及工作原理

    本文转载自:https://www.cnblogs.com/felixzh/p/5869212.html 1.ZooKeeper是什么?       ZooKeeper是一个分布式的,开放源码的分布式 ...

  8. mongo数据库的使用

    mongodb 是一个非关系型数据库,跟每一个数据库都没有关系,(mysql 是一个关系型数据库)他以集合(collections)问单位,他长得和 json 一样 mongo 数据库的下载,安装 自 ...

  9. centos7.2 安装nginx+php

    Nginx的安装 安装快速HTTP服务器“的Nginx”并配置HTTP服务器# install from EPEL [root@linuxprobe~]# yum --enablerepo=epel ...

  10. 依赖注入框架之dagger2

    主页: https://github.com/google/dagger 历史 * Dagger1是由Square公司受到Guice(https://github.com/google/guice)启 ...