#线段树,矩阵乘法#洛谷 7453 [THUSCH2017] 大魔法师
分析
首先考虑如果修改操作都是单点修改怎么做,
以第一种修改为例那么就是
\]
同理其它操作也能通过乘矩阵来维护,
考虑区间修改时询问为求和,那么只需要开懒标记即可
时间复杂度 \(O(4^3Q\log_2{n})\)
代码
#include <cstdio>
#include <cctype>
#define rr register
using namespace std;
const int N=250011,mod=998244353;
typedef long long lll; int n;
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
inline void print(int ans){
if (ans>9) print(ans/10);
putchar(ans%10+48);
}
struct maix{
lll p[4][4];
inline maix operator *(const maix &t)const{
rr maix _t;
for (rr int i=0;i<4;++i)
for (rr int j=0;j<4;++j){
_t.p[i][j]=0;
for (rr int k=0;k<4;++k)
_t.p[i][j]+=p[i][k]*t.p[k][j];
_t.p[i][j]%=mod;
}
return _t;
}
}u,lazy[N<<2];
struct Four{
lll p[4];
inline Four operator +(const Four &t)const{
rr Four _t;
for (rr int i=0;i<4;++i)
_t.p[i]=p[i]+t.p[i]>=mod?p[i]+t.p[i]-mod:p[i]+t.p[i];
return _t;
}
}ANS,w[N<<2];
inline bool Is_Unit(maix A){
for (rr int i=0;i<4;++i)
for (rr int j=0;j<4;++j)
if (A.p[i][j]!=u.p[i][j]) return 0;
return 1;
}
inline Four mul(Four A,maix B){
rr Four C;
for (rr int i=0;i<4;++i){
C.p[i]=0;
for (rr int j=0;j<4;++j)
C.p[i]+=A.p[j]*B.p[j][i];
C.p[i]%=mod;
}
return C;
}
inline void pdown(int k){
w[k<<1]=mul(w[k<<1],lazy[k]);
lazy[k<<1]=lazy[k<<1]*lazy[k];
w[k<<1|1]=mul(w[k<<1|1],lazy[k]);
lazy[k<<1|1]=lazy[k<<1|1]*lazy[k];
lazy[k]=u;
}
inline void build(int k,int l,int r){
lazy[k]=u;
if (l==r){
for (rr int i=0;i<3;++i)
w[k].p[i]=iut();
w[k].p[3]=1;
return;
}
rr int mid=(l+r)>>1;
build(k<<1,l,mid);
build(k<<1|1,mid+1,r);
w[k]=w[k<<1]+w[k<<1|1];
}
inline void update(int k,int l,int r,int x,int y,maix z){
if (l==x&&r==y) {w[k]=mul(w[k],z),lazy[k]=lazy[k]*z; return;}
rr int mid=(l+r)>>1; if (!Is_Unit(lazy[k])) pdown(k);
if (y<=mid) update(k<<1,l,mid,x,y,z);
else if (x>mid) update(k<<1|1,mid+1,r,x,y,z);
else update(k<<1,l,mid,x,mid,z),update(k<<1|1,mid+1,r,mid+1,y,z);
w[k]=w[k<<1]+w[k<<1|1];
}
inline Four query(int k,int l,int r,int x,int y){
if (l==x&&r==y) return w[k];
rr int mid=(l+r)>>1; if (!Is_Unit(lazy[k])) pdown(k);
if (y<=mid) return query(k<<1,l,mid,x,y);
else if (x>mid) return query(k<<1|1,mid+1,r,x,y);
else return query(k<<1,l,mid,x,mid)+query(k<<1|1,mid+1,r,mid+1,y);
}
signed main(){
for (rr int i=0;i<4;++i) u.p[i][i]=1;
n=iut(),build(1,1,n);
for (rr int Q=iut();Q;--Q){
rr int opt=iut(),l=iut(),r=iut(),w=0; rr maix z=u;
if (opt<7){
if (opt>3) w=iut();
switch (opt){
case 1:{
z.p[1][0]=1;
break;
}
case 2:{
z.p[2][1]=1;
break;
}
case 3:{
z.p[0][2]=1;
break;
}
case 4:{
z.p[3][0]=w;
break;
}
case 5:{
z.p[1][1]=w;
break;
}
case 6:{
z.p[2][2]=0,z.p[3][2]=w;
break;
}
}
update(1,1,n,l,r,z);
}
else{
ANS=query(1,1,n,l,r);
for (rr int i=0;i<3;++i)
print(ANS.p[i]),putchar(i==2?10:32);
}
}
return 0;
}
#线段树,矩阵乘法#洛谷 7453 [THUSCH2017] 大魔法师的更多相关文章
- 【Codeforces718C】Sasha and Array 线段树 + 矩阵乘法
C. Sasha and Array time limit per test:5 seconds memory limit per test:256 megabytes input:standard ...
- hdu 5068(线段树+矩阵乘法)
矩阵乘法来进行所有路径的运算, 线段树来查询修改. 关键还是矩阵乘法的结合律. Harry And Math Teacher Time Limit: 5000/3000 MS (Java/Others ...
- 【对不同形式矩阵的总结】WC 2009 最短路径问题(线段树+矩阵乘法)
题意 题目链接:https://www.luogu.org/problem/P4150 一个 \(6\times n\) 的网格图,每个格点有一个初始权值.有两种操作: 修改一个格子的权值 求 ...
- MAZE(2019年牛客多校第二场E题+线段树+矩阵乘法)
题目链接 传送门 题意 在一张\(n\times m\)的矩阵里面,你每次可以往左右和下三个方向移动(不能回到上一次所在的格子),\(1\)表示这个位置是墙,\(0\)为空地. 现在有\(q\)次操作 ...
- CF718C Sasha and Array 线段树 + 矩阵乘法
有两个操作: 将 $[l,r]$所有数 + $x$ 求 $\sum_{i=l}^{r}fib(i)$ $n=m=10^5$ 直接求不好求,改成矩阵乘法的形式: $a_{i}=M^x\times ...
- Wannafly Winter Camp Day8(Div1,onsite) E题 Souls-like Game 线段树 矩阵乘法
目录 Catalog Solution: (有任何问题欢迎留言或私聊 && 欢迎交流讨论哦 Catalog @ Problem:传送门 Portal 原题目描述在最下面. 简单的 ...
- LOJ2980 THUSC2017大魔法师(线段树+矩阵乘法)
线段树每个节点维护(A,B,C,len)向量,操作即是将其乘上一个矩阵. #include<iostream> #include<cstdio> #include<cma ...
- HDU 5068 Harry And Math Teacher 线段树+矩阵乘法
题意: 一栋楼有n层,每一层有2个门,每层的两个门和下一层之间的两个门之间各有一条路(共4条). 有两种操作: 0 x y : 输出第x层到第y层的路径数量. 1 x y z : 改变第x层 的 y门 ...
- [tsA1490][2013中国国家集训队第二次作业]osu![概率dp+线段树+矩阵乘法]
这样的题解只能舔题解了,,,qaq 清橙资料里有.. #include <iostream> #include <cstdio> #include <cstdlib> ...
- SP1716 GSS3(线段树+矩阵乘法)
Code: #include <bits/stdc++.h> #define N 50001 #define ll long long #define lson now<<1 ...
随机推荐
- 异常处理try...except...finally---day26
1.认识异常处理 # ### 认识异常处理 #IndexError 索引超出序列范围 #lst = [1,2,3,4] #print(lst[10]) #KeyError 字典中查找一个不存在的关键字 ...
- 【Azure Developer】使用Python代码获取VM的IP地址 (Public IP + Private IP)【未解决问题标签】
记录使用以下的代码获取Azure VM中的IP地址 """Create and manage virtual machines. This script expects ...
- 【Azure 应用服务】本地Git部署Java项目到App Server,访问无效的原因
问题描述 在App Server的部署中心配置好本地Git 仓库 并推送 git push azure master 分支代码到服务器时,并未发生错误 ,但是服务异常,无法访问到正确的项目文件,始终打 ...
- C# 多线程(17):小总结
前言 本篇内容是小总结和过渡,看完这篇后,就要开始继续学习 C# 多线程中的知识点啦~. 前面,经过 16 篇的学习,我们学习了多线程.锁.线程池.任务.同步.异步等知识,还没有使用到 async.a ...
- Java toString的使用
1 package com.bytezreo.objectclass; 2 3 import java.util.Date; 4 5 /** 6 * 7 * @Description Object类中 ...
- SPFA最短路
目录 从Bellman-Ford开始 核心思想 模拟算法执行过程 时间复杂度 模板 spfa spfa优化的思想 模板 从Bellman-Ford开始 对于所有边权都大于等于0的图,任意两个顶点之间的 ...
- Codeforces Round 729 (Div. 2)B. Plus and Multiply(构造、数学)
题面 链接 B. Plus and Multiply 题意 给定\(n,a,b\) 可以进行的操作 \(*a\) \(+b\) 最开始的数是1 问能否经过上面的两种操作将1变为n 题解 这题的关键是能 ...
- 【MongoDB详细步骤】(内附源码)
第01章-MongoDB 1.安装和启动(docker方式) 1.1.拉取镜像 docker pull mongo:4.4.8 1.2.创建和启动容器 docker run -d --restart= ...
- vue-simple-uploader 上传组件 用js调用 打开窗口上传
为什么有这个需求 需要弹框 让用户填些数据后,再进行上传,所以不能先点击上传按钮 重点1:添加id <uploader-btn :single="true" id=" ...
- 记本地新建一个gradle方式springboot项目过程
打算使用gradle在idea新建个springboot项目,然后坑很多,记录一下 原来我的idea应该是社区版,新建项目时候没有可以选择spring相关配置,然后卸载了重装,之前问题是启动是启动起来 ...