zoukankan      html  css  js  c++  java
  • 线段树模板3.0 区间乘

    线段树模板3.0 区间乘

    注意lazy标记应用

    先乘后加

    int MOD;
    
    struct Tree {
        ll l, r;
        ll sum;
        ll mul, add;
    };
    
    Tree node[maxn << 2];
    int a[maxn];
    
    void push_up(int i) {
        node[i].sum = node[i << 1].sum + node[i << 1 | 1].sum;
        node[i].sum %= MOD;
    }
    
    void build(int i, int l, int r) {
        node[i].l = l;
        node[i].r = r;
        node[i].mul = 1;
        if (l == r) {
            node[i].sum = a[l] % MOD;
            return;
        }
        int mid = l + r >> 1;
        build(i << 1, l, mid);
        build(i << 1 | 1, mid + 1, r);
        push_up(i);
    }
    
    void push_down(int i) {
        node[i << 1].sum = (ll)node[i].mul * node[i << 1].sum  % MOD + ((ll)(node[i << 1].r - node[i << 1].l + 1) * node[i].add % MOD) % MOD;
        node[i << 1].sum %= MOD;
        node[i << 1 | 1].sum = (ll)node[i].mul * node[i << 1 | 1].sum % MOD + ((ll)node[i].add * (node[i << 1 | 1].r - node[i << 1 | 1].l + 1) % MOD) % MOD;
        node[i << 1 | 1].sum %= MOD;
        
        node[i << 1].mul = (ll)node[i << 1].mul * node[i].mul % MOD;
        node[i << 1 | 1].mul = (ll)node[i << 1 | 1].mul * node[i].mul % MOD;
       
        node[i << 1].add = (ll)node[i << 1].add * node[i].mul % MOD + node[i].add;
        node[i << 1].add %= MOD;
        node[i << 1 | 1].add = (ll)node[i << 1 | 1].add * node[i].mul % MOD + node[i].add;
        node[i << 1 | 1].add %= MOD;
        
        node[i].mul = 1;
        node[i].add = 0;
    }
    
    void update1(int i, int l, int r, ll val) {
        if (node[i].l > r || node[i].r < l) return;
        if (node[i].l >= l && node[i].r <= r) {
            node[i].add += val;
            node[i].add %= MOD;
            node[i].sum += val * (node[i].r - node[i].l + 1) % MOD;
            node[i].sum %= MOD;
            return;
        }
        push_down(i);
        int mid = node[i].l + node[i].r >> 1;
        if(l <= mid) update1(i << 1, l, r, val);
        if(r > mid) update1(i << 1 | 1, l, r, val);
        push_up(i);
    }
    
    void update2(int i, int l, int r, ll val) {
        if (node[i].l > r || node[i].r < l) return;
        if (node[i].l >= l && node[i].r <= r) {
            node[i].add = (ll)node[i].add * val % MOD;    //注意这里add也要乘上val
            node[i].mul = (ll)node[i].mul * val % MOD;
            node[i].sum = (ll)node[i].sum * val % MOD;
            return;
        }
        push_down(i);
        update2(i << 1, l, r, val);
        update2(i << 1 | 1, l, r, val);
        push_up(i);
    }
    
    ll query(int i, int l, int r) {
        if (node[i].l > r || node[i].r < l) return 0;
        if (node[i].l >= l && node[i].r <= r)  return node[i].sum;
        push_down(i);
        return (query(i << 1, l, r) + query(i << 1 | 1, l, r)) % MOD;
    }
    
    int main() {
        int n = readint();
        int m = readint();
        MOD = readint();
        for (int i = 1; i <= n; i++)
            a[i] = readint();
        build(1, 1, n);
        for (int i = 1; i <= m; i++) {
            int op = readint();
            if (op == 1) {
                int L = readint();
                int R = readint();
                int V = readint();
                update2(1, L, R, V);
            }
            else if (op == 2) {
                int L = readint();
                int R = readint();
                int V = readint();
                update1(1, L, R, V);
            }
            else {
                int L = readint();
                int R = readint();
                Put(query(1, L, R));
                puts("");
            }
        }
    }
    
  • 相关阅读:
    素数路径Prime Path POJ3126 素数,BFS
    Fliptile POJ3279 DFS
    Find the Multiple POJ1426
    洗牌Shuffle'm Up POJ3087 模拟
    棋盘问题 POJ1321 DFS
    抓住那只牛!Catch That Cow POJ3278 BFS
    Dungeon Master POJ2251 三维BFS
    Splitting into digits CodeForce#1104A
    Ubuntu下手动安装Nvidia显卡驱动
    最大连续子序列和
  • 原文地址:https://www.cnblogs.com/hznumqf/p/13834513.html
Copyright © 2011-2022 走看看