BZOJ_1798_[AHOI2009]维护序列_线段树

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

分析:线段树上要打两个标记。要注意下传的顺序。显然先乘后加和先加后乘是不一样的。我们发现如果是先加后乘的话更改子树值的式子里会出现除法。不妨规定任何时候都先乘后加。推出的式子即为

t[lson]=(t[lson]*mul[pos]+add[pos]*(mid-l+1))%p;

mul[lson]=(mul[lson]*mul[pos])%p;

add[lson]=(add[lson]*mul[pos]+add[pos])%p;

代码:

 1 #include <stdio.h>
 2 #include <string.h>
 3 #include <algorithm>
 4 using namespace std;
 5 #define ls p<<1
 6 #define rs p<<1|1
 7 #define LL long long
 8 #define N 100050
 9 LL add[N<<2],t[N<<2],mul[N<<2],mod;
10 int a[N],n,m;
11 void bt(int l,int r,int p){
12     mul[p]=1;
13     if(l==r){
14         scanf("%lld",&t[p]);return ;
15     }
16     int mid=l+r>>1;
17     bt(l,mid,ls);bt(mid+1,r,rs);
18     t[p]=(t[ls]+t[rs])%mod;
19 }
20 void pud(int l,int r,int p){
21     int mid=l+r>>1;
22     if(add[p]==0&&mul[p]==1)return ;
23     t[ls]=(t[ls]*mul[p]+add[p]*(mid-l+1))%mod;
24     t[rs]=(t[rs]*mul[p]+add[p]*(r-mid))%mod;
25     mul[ls]=mul[ls]*mul[p]%mod;
26     mul[rs]=mul[rs]*mul[p]%mod;
27     add[ls]=(add[ls]*mul[p]+add[p])%mod;
28     add[rs]=(add[rs]*mul[p]+add[p])%mod;
29     mul[p]=1;add[p]=0;
30 }
31 void upad(int l,int r,int x,int y,int c,int p){
32     if(x<=l&&y>=r){
33         add[p]=(add[p]+c)%mod;
34         t[p]+=1ll*(r-l+1)*c;t[p]%=mod;
35         return ;
36     }
37     int mid=l+r>>1;
38     pud(l,r,p);
39     if(x<=mid)upad(l,mid,x,y,c,ls);
40     if(y>mid)upad(mid+1,r,x,y,c,rs);
41     t[p]=(t[ls]+t[rs])%mod;
42 }
43 void upmu(int l,int r,int x,int y,int c,int p){
44     if(x<=l&&y>=r){
45         mul[p]=mul[p]*c%mod;
46         add[p]=add[p]*c%mod;
47         t[p]=t[p]*c%mod;
48         return ;
49     }
50     pud(l,r,p);
51     int mid=l+r>>1;
52     if(x<=mid)upmu(l,mid,x,y,c,ls);
53     if(y>mid)upmu(mid+1,r,x,y,c,rs);
54     t[p]=(t[ls]+t[rs])%mod;
55 }
56 LL query(int l,int r,int x,int y,int p){
57     if(x<=l&&y>=r)return t[p];
58     int mid=l+r>>1;
59     LL re=0;
60     pud(l,r,p);
61     if(x<=mid)re=(re+query(l,mid,x,y,ls))%mod;
62     if(y>mid)re=(re+query(mid+1,r,x,y,rs))%mod;
63     return re;
64 }
65 int main(){
66     scanf("%d%lld",&n,&mod);
67     bt(1,n,1);
68     scanf("%d",&m);
69     int op,x,y,z;
70     for(int i=1;i<=m;i++){
71         scanf("%d",&op);
72         if(op==1){
73             scanf("%d%d%d",&x,&y,&z);
74             upmu(1,n,x,y,z,1);
75         }else if(op==2){
76             scanf("%d%d%d",&x,&y,&z);
77             upad(1,n,x,y,z,1);
78         }else{
79             scanf("%d%d",&x,&y);
80             printf("%lld\n",query(1,n,x,y,1));
81         }
82     }
83 }

原文地址:https://www.cnblogs.com/suika/p/8457042.html

时间: 2024-07-30 22:32:04

BZOJ_1798_[AHOI2009]维护序列_线段树的相关文章

BZOJ 1798 [Ahoi2009]维护序列seq (线段树)

题意 对于一个给定的序列有3种操作: 1.给一个区间的数乘c 2.给一个区间的数加c 3.查询区间和. 思路 就是普通的线段树区间更新,因为更新操作有两种,维护两个延迟标记就可以了,不过要注意乘和加在更新时相互之间的关系,在更新乘的时候之前加的数也要相应的乘,更新加的时候之前所乘的数没有改变. 代码 #include <stdio.h> #include <string.h> #include <iostream> #include <algorithm>

P2023 [AHOI2009] 维护序列(线段树水题)

题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一段数全部加一个值; (3)询问数列中的一段数的和,由于答案可能很大,你只需输出这个数模P的值. 输入输出格式 输入格式: 第一行两个整数N和P(1≤P≤1000000000).第二行含有N个非负整数,从左到右依次为a1,a2,…,aN, (0≤ai≤1000000000,1≤i≤N).第三行有一个整

P2023 [AHOI2009]维护序列 (线段树区间修改查询)

题目链接:https://www.luogu.org/problemnew/show/P2023 一道裸的线段树区间修改题,懒惰数组注意要先乘后加 #include<bits/stdc++.h> using namespace std; typedef long long LL; const int maxx = 400010; LL tree[maxx],lazy1[maxx],lazy2[maxx],a[maxx],mod; int n; void build(int l,int r,in

bzoj 1798: [Ahoi2009]Seq 维护序列seq 线段树 区间乘法区间加法 区间求和

1798: [Ahoi2009]Seq 维护序列seq Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/problem.php?id=1798 Description 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一段数全部加一个值; (3)询问数列

bzoj1798: [Ahoi2009]Seq 维护序列seq 线段树

题目传送门 这道题就是线段树 先传乘法标记再传加法 #include<cstdio> #include<cstring> #include<algorithm> #define LL long long using namespace std; const int M=400010; LL read(){ LL ans=0,f=1,c=getchar(); while(c<'0'||c>'9'){if(c=='-') f=-1; c=getchar();}

BZOJ 1798 [Ahoi2009]Seq 维护序列seq 线段树

题意:链接 方法:线段树 解析: 俩标记sb题 更新乘的时候更新加 完了 代码: #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #define lson l,mid,rt<<1 #define rson mid+1,r,rt<<1|1 #define N 100010 using namespace std; typedef

bzoj 1798 [Ahoi2009]Seq 维护序列seq ——线段树

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1798 先乘后加,就可给加法标记乘上乘法标记. 注意可能有 *0 的操作,所以 pshd 时不是 cg[ cr ]>1 而是 cg[ cr ]!=1 . #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #define ll long long #de

bzoj 1798 Seq 维护序列seq —— 线段树

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1798 这题还4A... 注意:cnt 从1开始:各种模 p:乘法标记初始值是 1:可能乘 0. 代码如下: #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #define mid ((l+r)>>1) using namespace std;

P2023 [AHOI2009]维护序列 题解(线段树

题目链接 P2023 [AHOI2009]维护序列 解题思路 线段树板子.不难,但是...有坑.坑有多深?一页\(WA\). 由于乘法可能乘\(k=0\),我这种做法可能会使结果产生负数.于是就有了这篇题解. (详情见代码注释) AC代码 #include<stdio.h> #define min(a,b) (a>b?b:a) #define max(a,b) (a>b?a:b) typedef long long ll; int n,m; ll mod,k,a[500010];