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 解题报告的更多相关文章

  1. 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)

    To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...

  2. 线段树_区间加乘(洛谷P3373模板)

    题目描述 如题,已知一个数列,你需要进行下面三种操作: 1.将某区间每一个数乘上x 2.将某区间每一个数加上x 3.求出某区间每一个数的和 输入格式: 第一行包含三个整数N.M.P,分别表示该数列数字 ...

  3. 洛谷 - P1198 - 最大数 - 线段树

    https://www.luogu.org/problemnew/show/P1198 要问区间最大值,肯定是要用线段树的,不能用树状数组.(因为没有逆元?但是题目求的是最后一段,可以改成类似前缀和啊 ...

  4. 洛谷 P2391 白雪皑皑 线段树+优化

    题目描述: 现在有 \(N\) 片雪花排成一列. Pty 要对雪花进行$ M $次染色操作,第 \(i\)次染色操作中,把\((i*p+q)%N+1\) 片雪花和第\((i*q+p)%N+1\)片雪花 ...

  5. 【洛谷】【线段树】P1471 方差

    [题目背景:] 滚粗了的HansBug在收拾旧数学书,然而他发现了什么奇妙的东西. [题目描述:] 蒟蒻HansBug在一本数学书里面发现了一个神奇的数列,包含N个实数.他想算算这个数列的平均数和方差 ...

  6. 【洛谷】【线段树】P1047 校门外的树

    [题目描述:] 某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米.我们可以把马路看成一个数轴,马路的一端在数轴0的位置,另一端在L的位置:数轴上的每个整数点,即0,1,2,……,L ...

  7. 【洛谷】【线段树】P1886 滑动窗口

    [题目描述:] 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. [输入格式:] 输入一共 ...

  8. 【洛谷】【线段树】P3353 在你窗外闪耀的星星

    [题目描述:] /* 飞逝的的时光不会模糊我对你的记忆.难以相信从我第一次见到你以来已经过去了3年.我仍然还生动地记得,3年前,在美丽的集美中学,从我看到你微笑着走出教室,你将头向后仰,柔和的晚霞照耀 ...

  9. 洛谷P5280 [ZJOI2019]线段树

      https://www.luogu.org/problemnew/show/P5280 省选的时候后一半时间开这题,想了接近两个小时的各种假做法,之后想的做法已经接近正解了,但是有一些细节问题理不 ...

随机推荐

  1. Scrapy爬取美女图片续集 (原创)

    上一篇咱们讲解了Scrapy的工作机制和如何使用Scrapy爬取美女图片,而今天接着讲解Scrapy爬取美女图片,不过采取了不同的方式和代码实现,对Scrapy的功能进行更深入的运用.(我的新书< ...

  2. C#课后小作业

    有关C#基础的练手 跟大家一起分享下 1.让用户输入一个100以内的数 打印1-100之间所有的数,用户输入的数除外 2.让用户输入一个100以内的数 打印1-这个数之间所有的数的和 3.使用一个fo ...

  3. JavaWeb(三十五)——使用JDBC处理Oracle大数据

    一.Oracle中大数据处理 在Oracle中,LOB(Large Object,大型对象)类型的字段现在用得越来越多了.因为这种类型的字段,容量大(最多能容纳4GB的数据),且一个表中可以有多个这种 ...

  4. php 操作 oracle lob 数据2

    CREATE SEQUENCE mylobs_id_seq    NOMINVALUE    NOMAXVALUE    NOCYCLE    CACHE 20    NOORDERINCREMENT ...

  5. fastCMS数据库相关操作类

    fastCMS针对数据库的操作有以下几个类: 1.[paging_Class]分页类 此类用于分页检索数据库内符合条件的记录 1) 支持百万级数据分页 2) 支持多种类型的SQL语法,比如 Left ...

  6. leetcode13_C++罗马数字转整数

    罗马数字包含以下七种字符: I, V, X, L,C,D 和 M. 字符 数值 I 1 V 5 X 10 L 50 C 100 D 500 M 1000 例如, 罗马数字 2 写做 II ,即为两个并 ...

  7. asp.net 设计条码code 11的问题

    前一段时间思考了一些条码生成的问题,其实条码也可以说是加密的文件显示. 一个条码首先要有规定 比如code 11 又 1234567890 - 这11个字符组成 而1 又用 5码 表示 "1 ...

  8. java 乐观锁 vs 悲观锁

    在数据库的锁机制中介绍过,数据库管理系统(DBMS)中的并发控制的任务是确保在多个事务同时存取数据库中同一数据时不破坏事务的隔离性和统一性以及数据库的统一性. 悲观锁其实就是 完全同步 比如 sync ...

  9. 第七次作业PSP

    psp 进度条 代码累积折线图 博文累积折线图 psp饼状图

  10. mysql数据库工具

    1.navicat12 中文版及破解 链接:https://pan.baidu.com/s/1TH8m6lduHJybUGhmjFPIAA 提取码:kwcd 2.旧版本mysql-front(连接可选 ...