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. Spring-boot简单的理解

    SpringBoot启动 SpringApplication.run(MyBootApplication.class); SpringApplication.run启动SpringBoot应用,主要过 ...

  2. js如何判断IE浏览器的版本包括IE11

    IE浏览器真是个坑:从ie6以及以前IE版本,简直就是垃圾,不按照Mozilla国际组织的标准来,乱搞.搞得兼容性很差:   <script type="text/javascript ...

  3. 【模板】埃拉托色尼筛法 && 欧拉筛法 && 积性函数

    埃拉托色尼筛法 朴素算法 1 vis[1]=1; 2 for (int i=2;i<=n;i++) 3 if (!vis[i]) 4 { 5 pri[++tot]=i; 6 for (int j ...

  4. 分布式理论基础(一)一致性及解决一致性的两种方式:2PC和3PC (转载 不错)

    分布式理论基础(一)一致性及解决一致性的两种方式:2PC和3PC 1 一致性 1.1 简述 一致性,是指对每个节点一个数据的更新,整个集群都知道更新,并且是一致的 假设一个具有N个节点的分布式系统,当 ...

  5. confirm(message)

    如果用户点击确定按钮,则 confirm() 返回 true.如果点击取消按钮,则 confirm() 返回 false. 在用户点击确定按钮或取消按钮把对话框关闭之前,它将阻止用户对浏览器的所有输入 ...

  6. 小tip: 某简单的字符重叠与图形生成----张鑫旭

    引言 字符重叠不是什么稀奇的东西. 如1像素错位模拟阴影效果: 或者powerFloat中展示的带边框三角: 以及其他很多. 但是技术这东西不是豆腐,老了可以吃,臭了也可以吃:那我这里还拿着个说事作甚 ...

  7. AE三维点击查询(3D Identify)的实现(转)

    AE三维点击查询(3D Identify)的实现,类似ArcGIS的Identify对话框/////////////////////////////////////////////////////// ...

  8. Tomcat启动中文乱码解决方法

    一:解决方法一: 1.查看电脑系统的编码 针对Windows平台下,点击运行--输入cmd,enter键进入命令窗口,输入:chcp可以得到操作系统的代码页信息(代码页:字符集编码的别名),可以从控制 ...

  9. koa 中,中间件异步与同步的相关问题

    同步中间件很容易理解,如以下代码: const Router = require('koa-router') , koa = new Router({ prefix: '/koa' }) , fs = ...

  10. OpenGL学习--05--纹理立方体--BMP文件格式详解(转载)

    http://blog.csdn.net/o_sun_o/article/details/8351037 BMP文件格式详解 BMP文件格式详解(BMP file format) BMP文件格式,又称 ...