Description

老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成。 有长为N的数列,不妨设为a1,a2,…,aN 。有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一段数全部加一个值; (3)询问数列中的一段数的和,由于答案可能很大,你只需输出这个数模P的值。

Input

第一行两个整数N和P(1≤P≤1000000000)。第二行含有N个非负整数,从左到右依次为a1,a2,…,aN, (0≤ai≤1000000000,1≤i≤N)。第三行有一个整数M,表示操作总数。从第四行开始每行描述一个操作,输入的操作有以下三种形式: 操作1:“1 t g c”(不含双引号)。表示把所有满足t≤i≤g的ai改为ai×c (1≤t≤g≤N,0≤c≤1000000000)。 操作2:“2 t g c”(不含双引号)。表示把所有满足t≤i≤g的ai改为ai+c (1≤t≤g≤N,0≤c≤1000000000)。 操作3:“3 t g”(不含双引号)。询问所有满足t≤i≤g的ai的和模P的值 (1≤t≤g≤N)。 同一行相邻两数之间用一个空格隔开,每行开头和末尾没有多余空格。

Output

对每个操作3,按照它在输入中出现的顺序,依次输出一行一个整数表示询问结果。

Sample Input

7 43
1 2 3 4 5 6 7
5
1 2 5 5
3 2 4
2 3 7 9
3 1 3
3 4 7

Sample Output

2
35
8

HINT

【样例说明】

初始时数列为(1,2,3,4,5,6,7)。
经过第1次操作后,数列为(1,10,15,20,25,6,7)。
对第2次操作,和为10+15+20=45,模43的结果是2。
经过第3次操作后,数列为(1,10,24,29,34,15,16}
对第4次操作,和为1+10+24=35,模43的结果是35。
对第5次操作,和为29+34+15+16=94,模43的结果是8。

测试数据规模如下表所示

数据编号 1 2 3 4 5 6 7 8 9 10
N= 10 1000 1000 10000 60000 70000 80000 90000 100000 100000
M= 10 1000 1000 10000 60000 70000 80000 90000 100000 100000

线段树模板题

 #include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
struct node
{
long long val;
long long mul;
long long add;
} Segt[+];
long long n,m,p,a[],INF; void build(long long node,long long a[],long long l,long long r)
{
Segt[node].add=;
Segt[node].mul=;//初始值要设正确
if (l==r)
Segt[node].val=a[l];
else
{
long long mid=(l+r)/;
build(node*,a,l,mid);
build(node*+,a,mid+,r);
Segt[node].val=(Segt[node*].val+Segt[node*+].val)%p;
}
} void Push(long long node,long long l,long long r)
{ if (Segt[node].mul!=)
{
//因为乘法可以影响区间的加法,但加法无法影响区间的乘法
//所以更新乘法的时候要把左右儿子的加法也乘上
Segt[node*].mul=(Segt[node*].mul*Segt[node].mul)%p;
Segt[node*+].mul=(Segt[node*+].mul*Segt[node].mul)%p;
Segt[node*].add=(Segt[node*].add*Segt[node].mul)%p;
Segt[node*+].add=(Segt[node*+].add*Segt[node].mul)%p; long long mid=(l+r)/;
Segt[node*].val=(Segt[node*].val*Segt[node].mul)%p;
Segt[node*+].val=(Segt[node*+].val*Segt[node].mul)%p;
Segt[node].mul=;
}
if (Segt[node].add!=)
{
Segt[node*].add=(Segt[node*].add+Segt[node].add)%p;
Segt[node*+].add=(Segt[node*+].add+Segt[node].add)%p;
long long mid=(l+r)/;
Segt[node*].val=(Segt[node*].val+Segt[node].add*(mid-l+))%p;
Segt[node*+].val=(Segt[node*+].val+Segt[node].add*(r-mid))%p;
Segt[node].add=;
}
} void MulUpdate(int node,int l,int r,int l1,int r1,int k)
{
if (r1<l || l1>r)
return;
if (l1<=l&&r<=r1)
{
Segt[node].val=(Segt[node].val*k)%p;
Segt[node].add=(Segt[node].add*k)%p;
Segt[node].mul=(Segt[node].mul*k)%p;
return;
}
Push(node,l,r);
long long mid=(l+r)/;
MulUpdate(node*,l,mid,l1,r1,k);
MulUpdate(node*+,mid+,r,l1,r1,k);
Segt[node].val=(Segt[node*].val+Segt[node*+].val)%p;
} void AddUpdate(long long node,long long l,long long r,long long l1,long long r1,long long k)
{
if (r1<l || l1>r)
return;
if (l1<=l&&r<=r1)
{
Segt[node].val=(Segt[node].val+k*(r-l+))%p;
Segt[node].add=(Segt[node].add+k)%p;
return;
}
Push(node,l,r);
long long mid=(l+r)/;
AddUpdate(node*,l,mid,l1,r1,k);
AddUpdate(node*+,mid+,r,l1,r1,k);
Segt[node].val=(Segt[node*].val+Segt[node*+].val)%p;
} long long Query(long long node,long long l,long long r,long long l1,long long r1)
{
if (l1>r||r1<l)
return ;
if (l1<=l&&r<=r1)
return Segt[node].val;
Push(node,l,r);
long long mid=(l+r)/;
return (Query(node*,l,mid,l1,r1)+Query(node*+,mid+,r,l1,r1))%p;
} int main()
{
long long x,y,k,h;
scanf("%lld%lld",&n,&p);
for (int i=; i<=n; ++i)
scanf("%lld",&a[i]);
scanf("%lld",&m);
build(,a,,n);
for (int i=; i<=m; ++i)
{
scanf("%lld",&h);
if (h==)
scanf("%lld%lld%lld",&x,&y,&k),MulUpdate(,,n,x,y,k);
if (h==)
scanf("%lld%lld%lld",&x,&y,&k),AddUpdate(,,n,x,y,k);
if (h==)
scanf("%lld%lld",&x,&y),printf("%lld\n",Query(,,n,x,y)%p);
}
}

1798. [AHOI2009]维护序列【线段树】的更多相关文章

  1. BZOJ1798[Ahoi2009]维护序列——线段树

    题目描述     老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成.    有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2 ...

  2. [P2023][AHOI2009]维护序列(线段树)

    题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一 ...

  3. [AHOI2009]维护序列 (线段树)

    题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,-,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一 ...

  4. 洛谷 P2023 [AHOI2009]维护序列 || 线段树加法和乘法运算

    原理倒是非常简单.设原数为x,加法的lazytag为b,乘法的lazytag为a,操作数为c,那么原式为ax+b,乘上c后(ax+b)c=(ac)*x+b*c,加上c后(ax+b)+c=ax+(b+c ...

  5. [BZOJ1798][AHOI2009]Seq维护序列 线段树

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1798 一眼看过去线段树,事实上就是线段树.对于乘和加的两个标记,我们可以规定一个顺序,比如 ...

  6. BZOJ 1798 AHOI2009 Seq 维护序列 线段树

    题目大意:维护一个序列,提供三种操作: 1.将区间中每个点的权值乘上一个数 2.将区间中每个点的权值加上一个数 3.求一段区间的和对p取模的值 2631的超^n级弱化版.写2631之前能够拿这个练练手 ...

  7. 【AHOI2009】 维护序列 - 线段树

    题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一 ...

  8. 洛谷 P2023 BZOJ 1798 [AHOI2009]维护序列

    题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一 ...

  9. 洛谷 P2023 维护序列——线段树

    先上一波题目 https://www.luogu.org/problem/P2023 复习了一波线段树 题目涉及的操作有区间加 区间乘以及区间求和 tips:线段树在传标记的时候 优先传乘法标记再传加 ...

随机推荐

  1. 啰哩吧嗦式讲解在windows 家庭版安装docker

    1.docker是什么,为什么要使用docker Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中, 然后发布到任何流行的 Linux 机器上,也可以实 ...

  2. 《码出高效 Java开发手册》第五章 异常与日志

    码云: https://gitee.com/forxiaoming/JavaBaseCode/blob/master/EasyCoding/src/exception/index.md 5.2 try ...

  3. spring boot获取request

    1. Controller中 1.1 通过静态方法获取 HttpServletRequest request = ((ServletRequestAttributes)RequestContextHo ...

  4. hdu 1885

    Key Task Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Su ...

  5. 可缺省的CSS布局——张鑫旭

    一.技术不难.意识很难 有些东西的东西的实现,难的不是原料.技术:而是想不到,或者说意识不到. 例如下面这个简单而又神奇的魔术: 是吧.搞通了,才发现,哦~原来这么回事,很简单的嘛,我也可以实现的!其 ...

  6. css 相对单位rem详解

      CSS3新增了一个相对单位rem(root em,根em),这个单位引起了广泛关注.这个单位与em有什么区别呢?区别在于使用rem为元素设定字体大小时,仍然是相对大小,但相对的只是HTML根元素. ...

  7. python3中的新式类mro查看和C3算法原理

    两个公式 L(object) = [object] L(子类(父类1, 父类2)) = [子类] + merge(L(父类1), L(父类2) , [父类1, 父类2])注意 + 代表合并列表 mer ...

  8. charles 抓包 https 证书

    1. 概述 环境:这里是windows8 和 android (参考了ios环境的博客) 手机app点击发出http及https的请求,之前抓包都有请求的相关内容展示,这次没有,原来之前的一直抓的是h ...

  9. springboot项目的重定向和转发

    下面是idea软件创建的项目目录,这里总结了一下转发与重定向的问题,详解如下. 首先解释一下每个文件夹的作用,如果你是用的是idea创建的springboot项目,会在项目创建的一开始resource ...

  10. 11招教你做好 ERP 系统维护

    ERP 维护的具体工作内容主要包括以下几个方面: 例行和突发事件的处理 以管理和技术的手段,维护和发展 ERP 运行环境,如平衡技术先进性/实用风险.目标/成本而进行的IT基础结构(服务器.网络.PC ...