洛谷P2023 [AHOI2009]维护序列
? 解题记录 ? ? 洛谷 ? ? 线段树 ?    2017-10-24 23:53:43    354    0    0

 

题目描述

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

输入输出格式

输入格式:

 

第一行两个整数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)。 同一行相邻两数之间用一个空格隔开,每行开头和末尾没有多余空格。

 

输出格式:

 

对每个操作3,按照它在输入中出现的顺序,依次输出一行一个整数表示询问结果。

 

输入输出样例

输入样例#1: 复制
  1. 7 43
  2. 1 2 3 4 5 6 7
  3. 5
  4. 1 2 5 5
  5. 3 2 4
  6. 2 3 7 9
  7. 3 1 3
  8. 3 4 7
输出样例#1: 复制
  1. 2
  2. 35
  3. 8

说明

【样例说明】

初始时数列为(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

Source: Ahoi 2009

裸的加乘线段树,用ZKW交了一发,快还是要快,但可能是因为姿势不对,没有显著效果。但是经过两次打板子,zkw的正确率有了很大的提升。下面是代码:

  1. #include<cstdio>
  2. #include<cmath>
  3. #include<cctype>
  4. #define LL long long
  5. using namespace std;
  6. const int maxn = 1.5e5 + 10;
  7. int n, m, bit;
  8. LL data[maxn], size[maxn << 2], tree[maxn << 2], adds[maxn << 2], times[maxn << 2], p;
  9. inline void read(int & x) {
  10.     x = 0;static char c = getchar();
  11.     while(!isdigit(c)) c = getchar();
  12.     while(isdigit(c)) x = (<< 1) + (<< 3) + c - '0', c = getchar();
  13. }
  14. inline void read(LL & x) {
  15.     x = 0;static char c = getchar();
  16.     while(!isdigit(c)) c = getchar();
  17.     while(isdigit(c)) x = (<< 1) + (<< 3) + c - '0', c = getchar();
  18. }
  19. void push_up(int rt) {
  20.     tree[rt] = (tree[rt << 1] + tree[rt << 1 | 1]) % p;
  21. }
  22. void add_node(int rt, LL x) {
  23.     (adds[rt] += x) %= p, (tree[rt] += size[rt] * x) %= p;
  24. }
  25. void times_node(int rt, LL x) {
  26.     (times[rt] *= x) %= p, (tree[rt] *= x) %= p, (adds[rt] *= x) %= p;
  27. }
  28. void build(int n) {
  29.     for(bit = 1; bit < n + 3; bit <<= 1);/*n:重点!重点!!!*/
  30.     for(register int i = 1; i <= n; ++i) tree[+ bit] = data[i] % p;
  31.     for(register int i = bit; i <= (bit << 1); ++i) times[i] = 1, size[i] = 1;
  32.     for(register int i = bit - 1; i; --i) 
  33.         size[i] = size[<< 1] + size[<< 1 | 1], times[i] = 1, push_up(i);
  34. }
  35. void push_down(int rt) {
  36.     if(rt ^ 1) push_down(rt >> 1);
  37.     if(times[rt] ^ 1) {
  38.         times_node(rt << 1, times[rt]);
  39.         times_node(rt << 1 | 1, times[rt]);
  40.     }
  41.     if(adds[rt]) {
  42.         add_node(rt << 1, adds[rt]);
  43.         add_node(rt << 1 | 1, adds[rt]);
  44.     }
  45.     times[rt] = 1, adds[rt] = 0;
  46. }
  47. void add(int l, int r, LL dt) {
  48.     push_down(bit + l - 1 >> 1), push_down(bit + r + 1 >> 1);
  49.     for(= bit + l - 1, r = bit + r + 1; l ^ r ^ 1; l >>= 1, r >>= 1) {
  50.         if(~& 1) add_node(^ 1, dt);
  51.         if(& 1) add_node(^ 1, dt);
  52.         push_up(>> 1), push_up(>> 1);
  53.     }
  54.     l >>= 1;
  55.     for(; l; l >>= 1) push_up(l);
  56. }
  57. void mtp(int l, int r, LL dt) {
  58.     push_down(bit + l - 1 >> 1), push_down(bit + r + 1 >> 1);
  59.     for(= bit + l - 1, r = bit + r + 1; l ^ r ^ 1; l >>= 1, r >>= 1) {
  60.         if(~& 1) times_node(^ 1, dt);
  61.         if(& 1) times_node(^ 1, dt);
  62.         push_up(>> 1), push_up(>> 1);
  63.     }
  64.     l >>= 1;
  65.     for(; l; l >>= 1) push_up(l);
  66. }
  67. LL query(int l, int r) {
  68.     LL ans = 0;
  69.     push_down(+ bit - 1 >> 1), push_down(+ bit + 1 >> 1);
  70.     for(= l + bit - 1, r = r + bit + 1; l ^ r ^ 1; l >>= 1, r >>= 1) {
  71.         if(~& 1)(ans += tree[^ 1]) %= p;
  72.         if(& 1)(ans += tree[^ 1]) %= p;
  73.     }
  74.     return ans;
  75. }
  76. int main() {
  77.     read(n), read(p);
  78.     for(register int i = 1; i <= n; ++i) read(data[i]);
  79.     build(n);
  80.     int o, a, b, c;
  81.     read(m);
  82.     while(m--) {
  83.         read(o);
  84.         if(== 1) read(a), read(b), read(c), mtp(a, b, c % p);
  85.         else if(== 2) read(a), read(b), read(c), add(a, b, c % p);
  86.         else read(a), read(b), printf("%lld\n", query(a, b));
  87.     }
  88.     return 0;
  89. }

 

 

 

上一篇: 洛谷P1290 欧几里德的游戏

下一篇: 洛谷P2320 [HNOI2006]鬼谷子的钱袋

354 人读过
立即登录, 发表评论.
没有帐号? 立即注册
0 条评论
文档导航