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. 什么是memcached?

    缓存是一种常驻与内存的内存数据库,内存的读取速度远远快于程序在磁盘读取数据的速度.我们在设计程序的时候常常会考虑使用缓存,将经常访问的数据放到内存上面这样可以提高访问数据的速度,同时可以降低磁盘或数据 ...

  2. [android] 手机卫士黑名单功能(列表展示)

    先把要拦截的电话号码保存到数据库中,拦截模式用个字段区分,1 电话拦截,2 短信拦截,3全部拦截 新建Activity类CallSmsSafeActivity.java 新建布局文件activity_ ...

  3. ubuntu 常见错误--Could not get lock /var/lib/dpkg/lock 问题修改

    ubuntu 常见错误–Could not get lock /var/lib/dpkg/lock 通过终端安装程序sudo apt-get install xxx或者apt-get update时出 ...

  4. @valid注解

    今天发现一个非常好用的注解直接上代码: 实体类domain: @Column(name = "NAME") @NotNull @Size(min = 2,max = 50) pri ...

  5. spring boot入门笔记 (一) - 一个简单的说明+一个案例

    spring boot 简化开发:把平时开发者最常用的到一些步骤,按照开发者的习惯,把能包装的就包装成一些固有的工具类(就比如我们之前连接数据库时常写的DB工具类).当然,是在原有的spring框架的 ...

  6. Spring学习手札(二)面向切面编程AOP

    AOP理解 Aspect Oriented Program面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术. 但是,这种说法有些片面,因为在软件工程中,AOP的价值体现的并 ...

  7. css-图文案例

    效果如下: 附上代码: <html> <head> <title>World</title> <style type="text/css ...

  8. vue-router 实现导航守卫(路由卫士)

    路由跳转前做一些验证,比如登录验证,是网站中的普遍需求. 对此,vue-route 提供的 beforeRouteUpdate 可以方便地实现导航守卫(navigation-guards). 导航守卫 ...

  9. 转:PHP中的使用curl发送请求(GET请求和POST请求)

    原文地址:http://www.jb51.net/article/104974.htm 使用CURL发送请求的基本流程 使用CURL的PHP扩展完成一个HTTP请求的发送一般有以下几个步骤: 1.初始 ...

  10. phoenix使用vue

    phoenix使用vue mix phoenix.new ass2 Fetch and install dependencies? [Yn] y 修改 package.json { "rep ...