bzoj1798 [Ahoi2009]维护序列
Description
Input
Output
Sample Input
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
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
正解:线段树。
这题不难,但是容易出错,所以还是挺有意义的。
我们除了维护一个加法的标记以外,还要维护一个乘法的标记,当我们下放更新乘法标记时,被更新的那个区间的加法标记也要乘上对应的数(这个应该很显然吧。。)。并且我们下放的时候,要先下放乘法标记,再下放加法标记。于是我们就能完美地解决这题了。
//It is made by wfj_2048~
#include <algorithm>
#include <iostream>
#include <complex>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <vector>
#include <cmath>
#include <queue>
#include <stack>
#include <map>
#include <set>
#define inf (1<<30)
#define ls (x<<1)
#define rs (x<<1|1)
#define N (100010)
#define il inline
#define RG register
#define ll long long
#define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout) using namespace std; ll sum[*N],m[*N],a[*N],n,Q,p; il ll gi(){
RG ll x=,q=; RG char ch=getchar();
while ((ch<'' || ch>'') && ch!='-') ch=getchar();
if (ch=='-') q=-,ch=getchar();
while (ch>='' && ch<='') x=x*+ch-,ch=getchar();
return q*x;
} il void build(RG ll x,RG ll l,RG ll r){
if (l==r){ sum[x]=gi(),m[x]=; return; }
RG ll mid=(l+r)>>;
build(ls,l,mid),build(rs,mid+,r);
sum[x]=sum[ls]+sum[rs]; m[x]=;
if (sum[x]>=p) sum[x]-=p; return;
} il void pushdown(RG ll x,RG ll l,RG ll r){
if (m[x]!=){
sum[ls]*=m[x],sum[ls]%=p;
sum[rs]*=m[x],sum[rs]%=p;
m[ls]*=m[x],m[ls]%=p;
m[rs]*=m[x],m[rs]%=p;
a[ls]*=m[x],a[ls]%=p;
a[rs]*=m[x],a[rs]%=p;
m[x]=;
}
if (a[x]!=){
RG ll mid=(l+r)>>;
sum[ls]+=(mid-l+)*a[x],sum[ls]%=p;
sum[rs]+=(r-mid)*a[x],sum[rs]%=p;
a[ls]+=a[x]; if (a[ls]>=p) a[ls]-=p;
a[rs]+=a[x]; if (a[rs]>=p) a[rs]-=p;
a[x]=;
}
return;
} il void mul(RG ll x,RG ll l,RG ll r,RG ll xl,RG ll xr,RG ll c){
if (xl<=l && r<=xr){
sum[x]*=c,sum[x]%=p;
m[x]*=c,m[x]%=p;
a[x]*=c,a[x]%=p;
return;
}
pushdown(x,l,r); RG ll mid=(l+r)>>;
if (xr<=mid) mul(ls,l,mid,xl,xr,c);
else if (xl>mid) mul(rs,mid+,r,xl,xr,c);
else mul(ls,l,mid,xl,mid,c),mul(rs,mid+,r,mid+,xr,c);
sum[x]=sum[ls]+sum[rs]; if (sum[x]>=p) sum[x]-=p; return;
} il void add(RG ll x,RG ll l,RG ll r,RG ll xl,RG ll xr,RG ll c){
if (xl<=l && r<=xr){
sum[x]+=(r-l+)*c,sum[x]%=p;
a[x]+=c; if (a[x]>=p) a[x]-=p;
return;
}
pushdown(x,l,r); RG ll mid=(l+r)>>;
if (xr<=mid) add(ls,l,mid,xl,xr,c);
else if (xl>mid) add(rs,mid+,r,xl,xr,c);
else add(ls,l,mid,xl,mid,c),add(rs,mid+,r,mid+,xr,c);
sum[x]=sum[ls]+sum[rs]; if (sum[x]>=p) sum[x]-=p; return;
} il ll query(RG ll x,RG ll l,RG ll r,RG ll xl,RG ll xr){
if (xl<=l && r<=xr) return sum[x];
pushdown(x,l,r); RG ll mid=(l+r)>>;
if (xr<=mid) return query(ls,l,mid,xl,xr);
else if (xl>mid) return query(rs,mid+,r,xl,xr);
else return (query(ls,l,mid,xl,mid)+query(rs,mid+,r,mid+,xr))%p;
} il void work(){
n=gi(),p=gi(); build(,,n); Q=gi();
for (RG ll i=,type,l,r,c;i<=Q;++i){
type=gi(),l=gi(),r=gi();
if (type==) c=gi(),mul(,,n,l,r,c);
if (type==) c=gi(),add(,,n,l,r,c);
if (type==) printf("%lld\n",query(,,n,l,r));
}
return;
} int main(){
File("seq");
work();
return ;
}
bzoj1798 [Ahoi2009]维护序列的更多相关文章
- 【线段树】Bzoj1798 [AHOI2009] 维护序列
Description 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2 ...
- BZOJ1798[Ahoi2009]维护序列——线段树
题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2 ...
- BZOJ_1798_[AHOI2009]维护序列_线段树
BZOJ_1798_[AHOI2009]维护序列_线段树 题意:老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: ( ...
- [Luogu 2023] AHOI2009 维护序列
[Luogu 2023] AHOI2009 维护序列 恕我冒昧这和线段树模板二有个琴梨区别? #include <cstdio> int n,m; long long p; class S ...
- [洛谷P2023] [AHOI2009]维护序列
洛谷题目链接:[AHOI2009]维护序列 题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,-,aN .有如下三种操作形式: (1)把数列 ...
- 洛谷 2023 [AHOI2009]维护序列
洛谷 2023 [AHOI2009]维护序列 洛谷原题传送门 这个题也是一道经典的线段树模版(其实洛谷的模版二改一下输入顺序就能AC),其中包括区间乘法修改.区间加法修改.区间查询三个操作. 线段树的 ...
- 洛谷 P2023 [AHOI2009]维护序列
P2023 [AHOI2009]维护序列 题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中 ...
- 洛谷 P2023 [AHOI2009]维护序列 题解
P2023 [AHOI2009]维护序列 题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,-,aN .有如下三种操作形式: (1)把数列中 ...
- 洛谷P2023 [AHOI2009]维护序列(线段树区间更新,区间查询)
洛谷P2023 [AHOI2009]维护序列 区间修改 当我们要修改一个区间时,要保证 \(ax+b\) 的形式,即先乘后加的形式.当将区间乘以一个数 \(k\) 时,原来的区间和为 \(ax+b\) ...
随机推荐
- mysql 修改数据库data存放位置
一.首先把mysql的服务先停掉. 二.更改MySQL配置文件My.ini中的数据库存储主路径 打开MySQL默认的安装文件夹C:\Program Files\MySQL\MySQL Server 5 ...
- synchronized的4种用法
1.方法声明时使用,放在范围操作符(public等)之后,返回类型声明(void等)之前.这时,线程获得的是成员锁,即一次只能有一个线程进入该方法,其他线程要想在此时调用该方法,只能排队等候,当前线程 ...
- Java 集合的理解(持续更新......)
一.集合的由来 通常,我们的程序需要根据程序运行时才知道创建多少个对象.但若非程序运行,程序开发阶段,我们根本不知道到底需要多少个数量的对象,甚至不知道它的准确类型.为了满足这些常规的编程需要,我们要 ...
- How to set up Tensorflow inception-v3 model on Windows
There is Inception-v3 model python implementation on GitHub at: https://github.com/tensorflow/models ...
- JavaSE之认识java
本来很早之前就应该总结自己在JavaSE中系统学到的知识了,马上就要出去工作了,想想自己还是非常菜的菜鸟,自己就夜不能寐呀.现在从zero基础开始带大家一起回顾学习的基础知识. 现在已经是凌晨了,但是 ...
- C++数据
const :常量 ~x == -++x == -(x+1) 二进制数,1变为0,0变为1 ^ 相同为0,不同为1 & ...
- 对Golang有兴趣的朋友,推荐一款go语言Web框架-dotweb
Go语言,2009年推出,对我个人,2015年下半年,才下定决心正式开始引入使用Go,自此,让我获得了一种全新的开发体验. 在不断的项目过程中,一个开发人员总喜欢堆积一些代码段,由于Go的开源特性,逐 ...
- 浅谈对java中传参问题的理解
之前用的c/c++比较多,在c/c++中对于传参类型,无外乎就是传值.传引用.传指针这几种.但在java中,由于没有指针类型,其传参的方式也发生了相应的变化.在网上找了找,按我之前的理解,java中传 ...
- ie11强制兼容模式打开
<meta http-equiv="X-UA-Compatible" content="IE=edge">
- Verilog读写文件
在通过编写Verilog代码实现ram功能时,需要自己先计算寄存器的位数和深度再编写代码. 而如果需要在编写的ram中预置值的话,就需要使用Verilog语言编写程序读写文件,来将相应的数据赋给寄存器 ...