题目大意:有$n$个数,有$m$个操作,有三种:

  1. $1\;l\;r\;x:$把区间$[l,r]$内的数乘上$x$
  2. $2\;l\;r\;x:$把区间$[l,r]$内的数加上$x$
  3. $3\;l\;r:$询问区间$[l,r]$的和,对$p$取模

(线段树2就是先读入$n\;m\;p$,再读入序列;本题是先读入$n\;p$,读入序列,再读入$m$,双倍经验)

题解:线段树,把$lazy\_tag$变成两个,分别记录区间加和区间乘,注意乘法的优先级比加法高

卡点:无(我以前写的是什么代码啊?)

C++ Code:

#include <cstdio>
#define maxn 100010 << 2
long long V[maxn], cov[maxn], tg[maxn];
int n, m;
int s[maxn], L, R;
long long p, x;
void update(int rt) {
V[rt] = (V[rt << 1] + V[rt << 1 | 1]) % p;
}
void build(int rt, int l, int r) {
cov[rt] = 1;
if (l == r) {
V[rt] = s[l] % p;
return ;
}
int mid = l + r >> 1;
build(rt << 1, l, mid);
build(rt << 1 | 1, mid + 1, r);
update(rt);
}
void pushdown(int rt, long long len) {
long long &Cov = cov[rt], &Tg = tg[rt];
V[rt << 1] = (V[rt << 1] * Cov + Tg * (len + 1 >> 1)) % p;
V[rt << 1 | 1] = (V[rt << 1 | 1] * Cov + Tg * (len >> 1)) % p;
cov[rt << 1] = (cov[rt << 1] * Cov) % p;
cov[rt << 1 | 1] = (cov[rt << 1 | 1] * Cov) % p;
tg[rt << 1] = (tg[rt << 1] * Cov + Tg) % p;
tg[rt << 1 | 1] = (tg[rt << 1 | 1] * Cov + Tg) % p;
Cov = 1, Tg = 0;
}
void add1(int rt, int l, int r) {
if (L <= l && R >= r) {
V[rt] = (V[rt] * x) % p;
cov[rt] = (cov[rt] * x) % p;
tg[rt] = (tg[rt] * x) % p;
return ;
}
int mid = l + r >> 1;
if (cov[rt] != 1 || tg[rt]) pushdown(rt, r - l + 1);
if (L <= mid) add1(rt << 1, l, mid);
if (R > mid) add1(rt << 1 | 1, mid + 1, r);
update(rt);
}
void add2(int rt, int l, int r) {
if (L <= l && R >= r) {
V[rt] = (V[rt] + x * (r - l + 1ll)) % p;
tg[rt] = (tg[rt] + x) % p;
return ;
}
int mid = l + r >> 1;
if (cov[rt] != 1 || tg[rt]) pushdown(rt, r - l + 1);
if (L <= mid) add2(rt << 1, l, mid);
if (R > mid) add2(rt << 1 | 1, mid + 1, r);
update(rt);
}
long long ask(int rt, int l, int r) {
if (L <= l && R >= r) return V[rt] % p;
int mid = l + r >> 1;
long long ans = 0;
if (cov[rt] != 1 || tg[rt]) pushdown(rt, r - l + 1);
if (L <= mid) ans = ask(rt << 1, l, mid);
if (R > mid) ans = (ans + ask(rt << 1 | 1, mid + 1, r)) % p;
return ans;
}
int main() {
scanf("%d%lld", &n, &p);
for (int i = 1; i <= n; i++) scanf("%d", s + i);
build(1, 1, n);
scanf("%d", &m);
while (m --> 0) {
long long op;
scanf("%lld%d%d", &op, &L, &R);
switch (op) {
case 1: {
scanf("%lld", &x);
add1(1, 1, n);
break;
}
case 2: {
scanf("%lld", &x);
add2(1, 1, n);
break;
}
default: printf("%lld\n", ask(1, 1, n));
}
}
return 0;
}

[bzoj1798][Ahoi2009]Seq 维护序列seq ([洛谷P3373]【模板】线段树 2)的更多相关文章

  1. 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)

    To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...

  2. BZOJ1798: [Ahoi2009]Seq 维护序列seq[线段树]

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 5504  Solved: 1937[Submit ...

  3. BZOJ 1798: [Ahoi2009]Seq 维护序列seq( 线段树 )

    线段树.. 打个 mul , add 的标记就好了.. 这个速度好像还挺快的...( 相比我其他代码 = = ) 好像是#35.. ---------------------------------- ...

  4. 1798: [Ahoi2009]Seq 维护序列seq

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 2930  Solved: 1087[Submit ...

  5. bzoj 1798: [Ahoi2009]Seq 维护序列seq (线段树 ,多重标记下放)

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 7773  Solved: 2792[Submit ...

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

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeO ...

  7. Bzoj 1798: [Ahoi2009]Seq 维护序列seq(线段树区间操作)

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec Memory Limit: 64 MB Description 老师交给小可可一个维护数列的任务,现在小可 ...

  8. BZOJ1798[Ahoi2009]Seq 维护序列seq 题解

    题目大意: 有长为N的数列,有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一段数全部加一个值; (3)询问数列中的一段数的和,由于答案可能很大,你只需输出这个数模P的值. ...

  9. 【bzoj1798】[Ahoi2009]Seq 维护序列seq 线段树

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

随机推荐

  1. vue项目中的函数封装

    项目中一般都会有fun.js这类的文件,里面有各种的如转换时间格式的,处理转换的等等函数方法. 其实经常用到的去获取基本数据的接口也可以封装成一个方法,方便复用. 如上面所标,获取列表数据之前需要先获 ...

  2. 洛谷P2759 奇怪的函数(log 二分)

    题目描述 使得 x^xxx 达到或超过 n 位数字的最小正整数 x 是多少? 输入输出格式 输入格式: 一个正整数 n 输出格式: 使得 x^xxx 达到 n 位数字的最小正整数 x 输入输出样例 输 ...

  3. 用servlet设计OA管理系统时遇到问题

    如果不加单引号会使得除变量和int类型的值不能传递 转发和重定向的区别 转发需要填写完整路径,重定向只需要写相对路径.原因是重定向是一次请求之内已经定位到了服务器端,转发则需要两次请求每次都需要完整的 ...

  4. Q&A - Nginx是做什么的?tomcat结合Nginx使用小结

    相信很多人都听过nginx,这个小巧的东西慢慢地在吞食apache和IIS的份额.那究竟它有什么作用呢?可能很多人未必了解. 说到反向代理,可能很多人都听说,但具体什么是反向代理,很多人估计就不清楚了 ...

  5. 使用inotify-tools与rsync构建实时备份系统

    使用inotifywait监控文件变动 inotifywait是 inotify-tools 包中提供的一个工具,它使用 inotify API 来监控文件/目录中的变动情况. 在archlinux上 ...

  6. SQL-批量插入和批量更新

    批量插入 表结构一样或类似 如果两张表的结构一样,例如一个表的结构和另一个表的结构一样,只是其中一张是临时表,而另一张表是存储数据的表,我们需要进行一次表的迁移的话,我们可以这样. insert in ...

  7. caioj:1093: 并查集2(scy的删边问题) C++

    题目描述 [题目描述] 读入一个无向图(可能含有多个连通分支),输出最多能删掉多少条边,而不改变这个图任意两点的连通性(原来连通的两个点依然连通,不连通的依然不连通). [输入格式] 第一行为图的顶点 ...

  8. CF961E Tufurama 树状数组

    E. Tufurama One day Polycarp decided to rewatch his absolute favourite episode of well-known TV seri ...

  9. urllib使用一

    urllib.urlopen()方法: 参数: 1.url(要访问的网页链接http:或者是本地文件file:) 2.data(如果有,就会由GET方法变为POST方法,提交的数据格式必须是appli ...

  10. 20145202马超 《Java程序设计》第二周学习总结

    一.课后练习题目 1.D 并没有给number赋值导致出现错误. 2.A 10/3相当于10=3*3+1: 10/3=3. 3.D 4.D 很明显,byte类型的数据的话300就会出现溢出的情况. 5 ...