分析

在线段树上用\(4 \times 4\)的矩阵打标记。

代码

#include <bits/stdc++.h>
#define rin(i,a,b) for(register int i=(a);i<=(b);++i)
#define irin(i,a,b) for(register int i=(a);i>=(b);--i)
#define trav(i,a) for(register int i=head[a];i;i=e[i].nxt)
typedef long long LL;
using std::cin;
using std::cout;
using std::endl; inline int read(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
return x*f;
} const int MAXN=250005;
const LL MOD=998244353;
int n,m,ql,qr;
LL a[MAXN],b[MAXN],c[MAXN];
struct Mat{
LL a[4][4];
Mat(){memset(a,0,sizeof a);}
inline LL* operator [] (int x){return a[x];}
inline friend Mat operator * (Mat x,Mat y){
Mat ret;
ret[0][0]=(ret[0][0]+x[0][0]*y[0][0])%MOD;
ret[0][1]=(ret[0][1]+x[0][0]*y[0][1])%MOD;
ret[0][2]=(ret[0][2]+x[0][0]*y[0][2])%MOD;
ret[0][3]=(ret[0][3]+x[0][0]*y[0][3])%MOD;
ret[0][0]=(ret[0][0]+x[0][1]*y[1][0])%MOD;
ret[0][1]=(ret[0][1]+x[0][1]*y[1][1])%MOD;
ret[0][2]=(ret[0][2]+x[0][1]*y[1][2])%MOD;
ret[0][3]=(ret[0][3]+x[0][1]*y[1][3])%MOD;
ret[0][0]=(ret[0][0]+x[0][2]*y[2][0])%MOD;
ret[0][1]=(ret[0][1]+x[0][2]*y[2][1])%MOD;
ret[0][2]=(ret[0][2]+x[0][2]*y[2][2])%MOD;
ret[0][3]=(ret[0][3]+x[0][2]*y[2][3])%MOD;
ret[0][0]=(ret[0][0]+x[0][3]*y[3][0])%MOD;
ret[0][1]=(ret[0][1]+x[0][3]*y[3][1])%MOD;
ret[0][2]=(ret[0][2]+x[0][3]*y[3][2])%MOD;
ret[0][3]=(ret[0][3]+x[0][3]*y[3][3])%MOD;
ret[1][0]=(ret[1][0]+x[1][0]*y[0][0])%MOD;
ret[1][1]=(ret[1][1]+x[1][0]*y[0][1])%MOD;
ret[1][2]=(ret[1][2]+x[1][0]*y[0][2])%MOD;
ret[1][3]=(ret[1][3]+x[1][0]*y[0][3])%MOD;
ret[1][0]=(ret[1][0]+x[1][1]*y[1][0])%MOD;
ret[1][1]=(ret[1][1]+x[1][1]*y[1][1])%MOD;
ret[1][2]=(ret[1][2]+x[1][1]*y[1][2])%MOD;
ret[1][3]=(ret[1][3]+x[1][1]*y[1][3])%MOD;
ret[1][0]=(ret[1][0]+x[1][2]*y[2][0])%MOD;
ret[1][1]=(ret[1][1]+x[1][2]*y[2][1])%MOD;
ret[1][2]=(ret[1][2]+x[1][2]*y[2][2])%MOD;
ret[1][3]=(ret[1][3]+x[1][2]*y[2][3])%MOD;
ret[1][0]=(ret[1][0]+x[1][3]*y[3][0])%MOD;
ret[1][1]=(ret[1][1]+x[1][3]*y[3][1])%MOD;
ret[1][2]=(ret[1][2]+x[1][3]*y[3][2])%MOD;
ret[1][3]=(ret[1][3]+x[1][3]*y[3][3])%MOD;
ret[2][0]=(ret[2][0]+x[2][0]*y[0][0])%MOD;
ret[2][1]=(ret[2][1]+x[2][0]*y[0][1])%MOD;
ret[2][2]=(ret[2][2]+x[2][0]*y[0][2])%MOD;
ret[2][3]=(ret[2][3]+x[2][0]*y[0][3])%MOD;
ret[2][0]=(ret[2][0]+x[2][1]*y[1][0])%MOD;
ret[2][1]=(ret[2][1]+x[2][1]*y[1][1])%MOD;
ret[2][2]=(ret[2][2]+x[2][1]*y[1][2])%MOD;
ret[2][3]=(ret[2][3]+x[2][1]*y[1][3])%MOD;
ret[2][0]=(ret[2][0]+x[2][2]*y[2][0])%MOD;
ret[2][1]=(ret[2][1]+x[2][2]*y[2][1])%MOD;
ret[2][2]=(ret[2][2]+x[2][2]*y[2][2])%MOD;
ret[2][3]=(ret[2][3]+x[2][2]*y[2][3])%MOD;
ret[2][0]=(ret[2][0]+x[2][3]*y[3][0])%MOD;
ret[2][1]=(ret[2][1]+x[2][3]*y[3][1])%MOD;
ret[2][2]=(ret[2][2]+x[2][3]*y[3][2])%MOD;
ret[2][3]=(ret[2][3]+x[2][3]*y[3][3])%MOD;
ret[3][0]=(ret[3][0]+x[3][0]*y[0][0])%MOD;
ret[3][1]=(ret[3][1]+x[3][0]*y[0][1])%MOD;
ret[3][2]=(ret[3][2]+x[3][0]*y[0][2])%MOD;
ret[3][3]=(ret[3][3]+x[3][0]*y[0][3])%MOD;
ret[3][0]=(ret[3][0]+x[3][1]*y[1][0])%MOD;
ret[3][1]=(ret[3][1]+x[3][1]*y[1][1])%MOD;
ret[3][2]=(ret[3][2]+x[3][1]*y[1][2])%MOD;
ret[3][3]=(ret[3][3]+x[3][1]*y[1][3])%MOD;
ret[3][0]=(ret[3][0]+x[3][2]*y[2][0])%MOD;
ret[3][1]=(ret[3][1]+x[3][2]*y[2][1])%MOD;
ret[3][2]=(ret[3][2]+x[3][2]*y[2][2])%MOD;
ret[3][3]=(ret[3][3]+x[3][2]*y[2][3])%MOD;
ret[3][0]=(ret[3][0]+x[3][3]*y[3][0])%MOD;
ret[3][1]=(ret[3][1]+x[3][3]*y[3][1])%MOD;
ret[3][2]=(ret[3][2]+x[3][3]*y[3][2])%MOD;
ret[3][3]=(ret[3][3]+x[3][3]*y[3][3])%MOD;
return ret;
}
inline friend Mat operator + (Mat x,Mat y){
Mat ret;
ret[0][0]=(x[0][0]+y[0][0])%MOD;
ret[0][1]=(x[0][1]+y[0][1])%MOD;
ret[0][2]=(x[0][2]+y[0][2])%MOD;
ret[0][3]=(x[0][3]+y[0][3])%MOD;
return ret;
}
inline bool check(){
if(a[0][0]!=1||a[1][1]!=1||a[2][2]!=1||a[3][3]!=1) return true;
if(a[0][1]||a[0][2]||a[0][3]||a[1][0]||a[1][2]||a[1][3]||a[2][0]||a[2][1]||a[2][3]||a[3][0]||a[3][1]||a[3][2]) return true;
return false;
}
}unit,optm[4],kk,seg[MAXN<<2],tag[MAXN<<2]; #define mid ((l+r)>>1)
#define lc (o<<1)
#define rc ((o<<1)|1) inline void pushtag(int o,Mat y){
seg[o]=seg[o]*y;
tag[o]=tag[o]*y;
} inline void pushdown(int o){
if(!tag[o].check()) return;
pushtag(lc,tag[o]);
pushtag(rc,tag[o]);
tag[o]=unit;
} void build(int o,int l,int r){
tag[o]=unit;
if(l==r){
seg[o][0][0]=a[l];
seg[o][0][1]=b[l];
seg[o][0][2]=c[l];
seg[o][0][3]=1;
return;
}
build(lc,l,mid);
build(rc,mid+1,r);
seg[o]=seg[lc]+seg[rc];
} void upd(int o,int l,int r){
if(ql<=l&&r<=qr){pushtag(o,kk);return;}
pushdown(o);
if(mid>=ql) upd(lc,l,mid);
if(mid<qr) upd(rc,mid+1,r);
seg[o]=seg[lc]+seg[rc];
} Mat query(int o,int l,int r){
if(ql<=l&&r<=qr) return seg[o];
pushdown(o);
if(mid<ql) return query(rc,mid+1,r);
else if(mid>=qr) return query(lc,l,mid);
else return query(lc,l,mid)+query(rc,mid+1,r);
} #undef mid
#undef lc
#undef rc int main(){
rin(i,0,3) unit[i][i]=1;
rin(i,1,3) optm[i]=unit;
optm[1][1][0]=1;
optm[2][2][1]=1;
optm[3][0][2]=1;
n=read();
rin(i,1,n) a[i]=read(),b[i]=read(),c[i]=read();
build(1,1,n);
m=read();
while(m--){
int opt=read();ql=read(),qr=read();
if(opt<=3) kk=optm[opt];
else if(opt==4) kk=unit,kk[3][0]=read();
else if(opt==5) kk=unit,kk[1][1]=read();
else if(opt==6) kk=unit,kk[2][2]=0,kk[3][2]=read();
else{
Mat ans=query(1,1,n);
printf("%lld %lld %lld\n",ans[0][0],ans[0][1],ans[0][2]);
continue;
}
upd(1,1,n);
}
return 0;
}

[THUSC2017]大魔法师:线段树的更多相关文章

  1. [LOJ#2980][THUSCH2017]大魔法师(线段树+矩阵)

    每个线段树维护一个行向量[A,B,C,len]分别是这个区间的A,B,C区间和与区间长度,转移显然. 以及此题卡常,稍微哪里写丑了就能100->45. #include<cstdio> ...

  2. LOJ 2980 「THUSCH 2017」大魔法师——线段树

    题目:https://loj.ac/problem/2980 线段树维护矩阵. 然后是 30 分.似乎是被卡常了?…… #include<cstdio> #include<cstri ...

  3. LOJ2980 THUSC2017大魔法师(线段树+矩阵乘法)

    线段树每个节点维护(A,B,C,len)向量,操作即是将其乘上一个矩阵. #include<iostream> #include<cstdio> #include<cma ...

  4. 【BZOJ-4530】大融合 线段树合并

    4530: [Bjoi2014]大融合 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 280  Solved: 167[Submit][Status] ...

  5. [BZOI2014]大融合——————线段树进阶

    竟然改了不到一小时就改出来了, 可喜可贺 Description Solution 一开始想的是边两侧简单路径之和的乘积,之后发现这是个树形结构,简单路径数就是节点数. 之后的难点就变成了如何求线段树 ...

  6. zhw大神线段树姿势

    ; i<; i++) tree[i][]=tree[i][]=i; ; i>=; i--) tree[i][]=tree[i+i][], tree[i][]=tree[i+i+][]; v ...

  7. HDU 5091---Beam Cannon(线段树+扫描线)

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=5091 Problem Description Recently, the γ galaxies bro ...

  8. HDU1199 动态线段树 // 离散化

    附动态线段树AC代码 http://acm.hdu.edu.cn/showproblem.php?pid=1199 因为昨天做了一道动态线段树的缘故,今天遇到了这题没有限制范围的题就自然而然想到了动态 ...

  9. [CodeChef - STREETTA] The Street 李超线段树

    大致题意: 给出两个序列A,B,A初始为负无穷,B初始为0,有三种操作 1.在A上区间[u,v]上加一个等差数列,取与原本A序列的最大值. 2.在B上区间[u,v]上加一个等差数列. 3.给出一个点X ...

随机推荐

  1. IIS服务搭建 试图加载格式不正确的程序

    1.基础步骤 https://jingyan.baidu.com/article/fedf073770f23335ac8977b1.html 2.错误解决 试图加载格式不正确的程序   解决:在IIS ...

  2. chrome://inspect 白屏BUG解决方案

    chrome://inspect 是前端常用的调试工具但是有时候会碰到PC端dev-tools白屏的问题 1.首先我们要处于[科][学]的网络环境 2.在 chrome 中输入chrome://app ...

  3. PythonWeb框架Django搭建过程

    首先下载PyCharm专业版 破解地址:https://www.52pojie.cn/thread-997094-1-1.html 之后创建python虚拟环境(创建虚拟环境在上一篇博客) 激活虚拟环 ...

  4. Python基础学习——文件操作、函数

    一.文件操作 文件操作链接:http://www.cnblogs.com/linhaifeng/articles/5984922.html(更多内容见此链接) 一.对文件操作流程 打开文件,得到文件句 ...

  5. Jpa 重写方言dialect 使用oracle / mysql 数据库自定义函数

    在使用criteria api进行查询时 criteriaBuilder只提供了一个部分标准的sql函数,但当我们需要使用oracle特有的行转列函数wm_concat或 mysql特有的行转列函数g ...

  6. npm学习(二)之如何防止权限错误

    如何防止权限错误 如果您在尝试全局安装包时看到EACCES错误,请阅读本章.如果更改安装npm的目录,通常可以避免此错误.要做到这一点,要么使用版本管理器重新安装npm(推荐)或手动更改npm的默认目 ...

  7. 091、万能的数据收集器 Fluentd (2019-05-15 周三)

    参考https://www.cnblogs.com/CloudMan6/p/7798224.html   前面的ELK 中我们使用的是 Filebeat 收集Docker日志,利用的是默认的loggi ...

  8. vue.js(4)--字符串跑马灯

    制作一个字符串的跑马灯效果 (1)实例代码 <!DOCTYPE html> <html lang="en"> <head> <meta c ...

  9. PAT Basic 1071 小赌怡情 (15 分)

    常言道“小赌怡情”.这是一个很简单的小游戏:首先由计算机给出第一个整数:然后玩家下注赌第二个整数将会比第一个数大还是小:玩家下注 t 个筹码后,计算机给出第二个数.若玩家猜对了,则系统奖励玩家 t 个 ...

  10. Codeforces Round #575 (Div. 3) B. Odd Sum Segments (构造,数学)

    B. Odd Sum Segments time limit per test3 seconds memory limit per test256 megabytes inputstandard in ...