洛谷 P3373 【模板】线段树 2 解题报告
P3373 【模板】线段树 2
题目描述
如题,已知一个数列,你需要进行下面三种操作:
1.将某区间每一个数乘上\(x\)
2.将某区间每一个数加上\(x\)
3.求出某区间每一个数的和
输入输出格式
输入格式:
第一行包含三个整数\(N\)、\(M\)、\(P\),分别表示该数列数字的个数、操作的总个数和模数。
第二行包含\(N\)个用空格分隔的整数,其中第\(i\)个数字表示数列第\(i\)项的初始值。
接下来\(M\)行每行包含3或4个整数,表示一个操作,具体如下:
操作1: 格式:1 \(x\) \(y\) \(k\) 含义:将区间\([x,y]\)内每个数乘上\(k\)
操作2: 格式:2 \(x\) \(y\) \(k\) 含义:将区间\([x,y]\)内每个数加上\(k\)
操作3: 格式:3 \(x\) \(y\) 含义:输出区间\([x,y]\)内每个数的和对\(P\)取模所得的结果
输出格式:
输出包含若干行整数,即为所有操作3的结果。
说明
时空限制:1000ms,128M
好题。
我们分别维护两个\(lazytag\),分别表示乘法和加法优先
不妨规定乘法优先,加法滞后
意思就是当我们\(pushdown\)时,总是对原区间先进行乘法操作,再进行加法操作
这样对于加法修改,我们直接做不用管乘法
对于乘法操作,我们直接对加法和乘法的\(lazytag\)做同样的操作即可
对于询问我们都给\(pushdown\)掉
Code:
#include <cstdio>
#define ls id<<1
#define rs id<<1|1
#define ll long long
const int N=100010;
ll lazymul[N<<2],lazyadd[N<<2],sum[N<<2],a[N];
ll n,m,p;
void push_downadd(ll id,ll L,ll R)
{
if(L!=R)
{
ll mid=L+R>>1;
sum[ls]=(sum[ls]+(mid+1-L)*lazyadd[id])%p;
sum[rs]=(sum[rs]+(R-mid)*lazyadd[id])%p;
(lazyadd[ls]+=lazyadd[id])%=p;
(lazyadd[rs]+=lazyadd[id])%=p;
}
lazyadd[id]=0;
}
void push_downmul(ll id,ll L,ll R)
{
if(lazymul[id]==1) return;
if(L!=R)
{
ll mid=L+R>>1;
sum[ls]=(sum[ls]*lazymul[id])%p;
sum[rs]=(sum[rs]*lazymul[id])%p;
(lazymul[ls]*=lazymul[id])%=p;
(lazymul[rs]*=lazymul[id])%=p;
(lazyadd[ls]*=lazymul[id])%=p;
(lazyadd[rs]*=lazymul[id])%=p;
}
lazymul[id]=1;
}
void updata(ll id)
{
sum[id]=(sum[ls]+sum[rs])%p;
}
void changemul(ll id,ll l,ll r,ll L,ll R,ll mult)
{
push_downmul(id,L,R);
push_downadd(id,L,R);
if(l==L&&r==R)
{
sum[id]=(sum[id]*mult)%p;
lazymul[id]=(lazymul[id]*mult)%p;
return;
}
ll mid=L+R>>1;
if(r<=mid) changemul(ls,l,r,L,mid,mult);
else if(l>mid) changemul(rs,l,r,mid+1,R,mult);
else changemul(ls,l,mid,L,mid,mult),changemul(rs,mid+1,r,mid+1,R,mult);
updata(id);
}
void changeadd(ll id,ll l,ll r,ll L,ll R,ll delta)
{
push_downmul(id,L,R);
push_downadd(id,L,R);
if(l==L&&r==R)
{
sum[id]=(sum[id]+(R+1-L)*delta)%p;
lazyadd[id]=(lazyadd[id]+delta)%p;
return;
}
ll mid=L+R>>1;
if(r<=mid) changeadd(ls,l,r,L,mid,delta);
else if(l>mid) changeadd(rs,l,r,mid+1,R,delta);
else changeadd(ls,l,mid,L,mid,delta),changeadd(rs,mid+1,r,mid+1,R,delta);
updata(id);
}
ll query(ll id,ll l,ll r,ll L,ll R)
{
if(l==L&&r==R)
return sum[id];
push_downmul(id,L,R);
push_downadd(id,L,R);
ll mid=L+R>>1;
if(r<=mid) return query(ls,l,r,L,mid);
else if(l>mid) return query(rs,l,r,mid+1,R);
else return (query(ls,l,mid,L,mid)+query(rs,mid+1,r,mid+1,R))%p;
}
void build(ll id,ll l,ll r)
{
lazymul[id]=1;
if(l==r)
{
sum[id]=a[l];
return;
}
ll mid=l+r>>1;
build(ls,l,mid);
build(rs,mid+1,r);
updata(id);
}
int main()
{
scanf("%lld%lld%lld",&n,&m,&p);
ll opt,x,y,k;
for(int i=1;i<=n;i++)
scanf("%lld",a+i);
build(1,1,n);
for(int i=1;i<=m;i++)
{
scanf("%lld%lld%lld",&opt,&x,&y);
if(opt==1)
{
scanf("%lld",&k);
changemul(1,x,y,1,n,k);
}
else if(opt==2)
{
scanf("%lld",&k);
changeadd(1,x,y,1,n,k);
}
else
printf("%lld\n",query(1,x,y,1,n));
}
return 0;
}
2018.7.25
洛谷 P3373 【模板】线段树 2 解题报告的更多相关文章
- 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)
To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...
- 线段树_区间加乘(洛谷P3373模板)
题目描述 如题,已知一个数列,你需要进行下面三种操作: 1.将某区间每一个数乘上x 2.将某区间每一个数加上x 3.求出某区间每一个数的和 输入格式: 第一行包含三个整数N.M.P,分别表示该数列数字 ...
- 洛谷 - P1198 - 最大数 - 线段树
https://www.luogu.org/problemnew/show/P1198 要问区间最大值,肯定是要用线段树的,不能用树状数组.(因为没有逆元?但是题目求的是最后一段,可以改成类似前缀和啊 ...
- 洛谷 P2391 白雪皑皑 线段树+优化
题目描述: 现在有 \(N\) 片雪花排成一列. Pty 要对雪花进行$ M $次染色操作,第 \(i\)次染色操作中,把\((i*p+q)%N+1\) 片雪花和第\((i*q+p)%N+1\)片雪花 ...
- 【洛谷】【线段树】P1471 方差
[题目背景:] 滚粗了的HansBug在收拾旧数学书,然而他发现了什么奇妙的东西. [题目描述:] 蒟蒻HansBug在一本数学书里面发现了一个神奇的数列,包含N个实数.他想算算这个数列的平均数和方差 ...
- 【洛谷】【线段树】P1047 校门外的树
[题目描述:] 某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米.我们可以把马路看成一个数轴,马路的一端在数轴0的位置,另一端在L的位置:数轴上的每个整数点,即0,1,2,……,L ...
- 【洛谷】【线段树】P1886 滑动窗口
[题目描述:] 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. [输入格式:] 输入一共 ...
- 【洛谷】【线段树】P3353 在你窗外闪耀的星星
[题目描述:] /* 飞逝的的时光不会模糊我对你的记忆.难以相信从我第一次见到你以来已经过去了3年.我仍然还生动地记得,3年前,在美丽的集美中学,从我看到你微笑着走出教室,你将头向后仰,柔和的晚霞照耀 ...
- 洛谷P5280 [ZJOI2019]线段树
https://www.luogu.org/problemnew/show/P5280 省选的时候后一半时间开这题,想了接近两个小时的各种假做法,之后想的做法已经接近正解了,但是有一些细节问题理不 ...
随机推荐
- Unity商店下载的文件保存路径?
Win7系统: C:\Users\系统用户名\AppData\Roaming\Unity\Asset Store MAC:"~/Library/Unity/Asset\ Store" ...
- MQ配置安装
一,MQ安装 ./mqlicense.sh -accept rpm -ivh MQSeries*.rpm -- rpm -qa|grep MQSeries 二,MQ配置 环境变量配置(MQM)实际安 ...
- CentOS 6.8 安装JDK8
JDK安装 1.查看环境是否有默认jdk,输入命令: rpm -qa | grep jdk 如果有默认jdk,可以使用 yum remove 删除 2.进入系统根目录,创建developer文件夹 3 ...
- 用树莓派Raspberry Pi和Micro:bit做一个自拍器
在这个项目中,我们将使用Python来构建一个由Micro:bit触发树莓派Raspberry Pi和相机模块的自拍器.这是开始使用硬件和简单文本编程的好方法. 我们将学习: 如何设置Raspberr ...
- 天马行空DevOps-Dev平台建设概述
概述 DevOps(Development和Operations的组合词)是一组过程.方法与系统的统称,用于促进开发(应用程序/软件工程).技术运营和质量保障(QA)部门之间的沟通.协作与整合.它是一 ...
- git blame 查看某行代码提交记录
1. 在当前git项目目录下执行 git blame -L 38,38 <filename> 例子: git blame -L 38,38 src/component/BarCode/i ...
- c++ Dynamic Memory (part 1)
1. make_shared<T>(args): return a shared_ptr dynamically allocated object of type T. Use args ...
- 0421--"数字口袋精灵app"二次开发(Blackbriar团队开发)
"数字口袋精灵app"二次开发 目录: 一.项目github总仓库推送 二.开发成员 三.分工与合作 四.各模块成果 五.心得墙 六.团队成员贡献分 内容: 一.项目github总 ...
- 上午做的第一个安卓app
刚开始学习安卓开发,有好多不懂,好多快捷键不知道,好多文件也不知道是干什么用的,初学时的确会有很多烦恼,比如哪里又多一个空格也会报错,有时候错误很难看懂. 嘿嘿,一上午的功夫边学习边调代码,做出了我第 ...
- “来用”Beta版使用说明
补发Beta版使用说明.Beta版与alpha版相比去掉了计算器,界面上没有太大变化. 1引言 1 .1编写目的 针对我们发布的Beta版本做出安装和使用说明,使参与内测的人员及用户了解软件的使用方法 ...