洛谷 P3384 【模板】树链剖分 题解

一、题目:

洛谷原题

二、代码:

//树链剖分模板,没啥思路可讲。
#include
#include

using namespace std;
inline int read(void) {
    int x = 0, f = 1; char ch = getchar();
    while (ch<'0' || ch>'9') { if (ch == '-')f = -1; ch = getchar(); }
    while (ch >= '0'&&ch <= '9') { x = x * 10 + ch - '0'; ch = getchar(); }
    return f * x;
}

const int maxn = 100005;

int n, m, root, mod, val[maxn], head[maxn], tot, id[maxn], cnt;

struct Edge {
    int y, next;
    Edge() {}
    Edge(int _y, int _next) :y(_y), next(_next) {}
}e[maxn << 1];

struct shu_lian_pou_fen {
    struct Node {
        int val, siz, heavy, mp, dep, top, fa;
        Node() { val = siz = heavy = mp = dep = fa = 0; }
    }t[maxn];
    inline void dfs1(int u, int fa, int deep) {
        t[u].val = val[u]; t[u].dep = deep; t[u].fa = fa;
        t[u].siz = 1;
        int maxson = -1;
        for (register int i = head[u]; i; i = e[i].next) {
            int v = e[i].y;
            if (v == fa)continue;
            dfs1(v, u, deep + 1); 
            t[u].siz += t[v].siz;
            if (maxson < t[v].siz) { t[u].heavy = v; maxson = t[v].siz; }
        }
    }
    inline void dfs2(int u, int first) {
        t[u].top = first;
        t[u].mp = ++cnt;
        id[cnt] = u;
        if (!t[u].heavy)return;
        dfs2(t[u].heavy, first);
        for (register int i = head[u]; i; i = e[i].next) {
            int v = e[i].y;
            if (v == t[u].fa || v == t[u].heavy)continue;
            dfs2(v, v);
        }
    }
}T;

struct xian_duan_shu {
#define lson (o<<1)
#define rson (o<<1|1)
    struct Node {
        int add, sum, len;
    }t[maxn << 2];
    inline void pushup(int o) {
        t[o].sum = (t[lson].sum + t[rson].sum) % mod;
    }
    inline void build(int o, int l, int r) {
        t[o].len = r - l + 1;
        if (l == r) { t[o].sum = val[id[l]]; return; }
        int mid = (l + r) >> 1;
        build(lson, l, mid);
        build(rson, mid + 1, r);
        pushup(o);
    }
    inline void pushdown(int o, int l, int r) {
        if (!t[o].add)return;
        (t[lson].add += t[o].add) %= mod; (t[lson].sum += t[lson].len*t[o].add) %= mod;
        (t[rson].add += t[o].add) %= mod; (t[rson].sum += t[rson].len*t[o].add) %= mod;
        t[o].add = 0;
    }
    inline void Update(int o, int l, int r, int ql, int qr, int v) {
        if (ql <= l && r <= qr) { t[o].add += v; t[o].sum += t[o].len*v; return; }
        pushdown(o, l, r);
        int mid = (l + r) >> 1;
        if (ql <= mid)Update(lson, l, mid, ql, qr, v);
        if (qr > mid)Update(rson, mid + 1, r, ql, qr, v);
        pushup(o);
    }
    inline int query(int o, int l, int r, int ql, int qr) {
        if (ql <= l && r <= qr) { return t[o].sum; }
        pushdown(o, l, r);
        int mid = (l + r) >> 1, ans = 0;
        if (ql <= mid)(ans += query(lson, l, mid, ql, qr)) %= mod;
        if (qr > mid)(ans += query(rson, mid + 1, r, ql, qr)) %= mod;
        return ans % mod;
    }
}TT;

inline int QUERY(int x, int y) {
    int ans = 0;
    // if(T.t[x].dep
    while (T.t[x].top != T.t[y].top) {
        if (T.t[T.t[x].top].dep < T.t[T.t[y].top].dep)x ^= y ^= x ^= y;
        (ans += TT.query(1, 1, n, T.t[T.t[x].top].mp, T.t[x].mp)) %= mod;
        x = T.t[T.t[x].top].fa;
    }
    if (T.t[x].dep > T.t[y].dep)x ^= y ^= x ^= y;
    (ans += TT.query(1, 1, n, T.t[x].mp, T.t[y].mp)) %= mod;
    return ans;
}

inline void UPDATE(int x, int y, int z) {
    z %= mod;
    while (T.t[x].top != T.t[y].top) {
        if (T.t[T.t[x].top].dep < T.t[T.t[y].top].dep)x ^= y ^= x ^= y;
        TT.Update(1, 1, n, T.t[T.t[x].top].mp, T.t[x].mp, z);
        x = T.t[T.t[x].top].fa;
    }
    if (T.t[x].dep>T.t[y].dep)x ^= y ^= x ^= y;
    TT.Update(1, 1, n, T.t[x].mp, T.t[y].mp, z);
}

inline void connect(int x, int y) {
    e[++tot] = Edge(y, head[x]); head[x] = tot;
}

int main() {
    freopen("out.txt", "w", stdout);
    n = read(); m = read(); root = read(); mod = read();
    for (register int i = 1; i <= n; i++) {
        val[i] = read();
    }
    for (register int i = 1; i <= n - 1; i++) {
        int x = read(), y = read();
        connect(x, y); connect(y, x);
    }
    T.dfs1(root, 0, 1);
    T.dfs2(root, root);
    TT.build(1, 1, n);
    for (register int i = 1; i <= m; i++) {
        int opt = read();
        if (opt == 1) {
            int x = read(), y = read(), z = read();
            // if(T.t[x].mp>T.t[y].mp)x^=y^=x^=y;
            UPDATE(x, y, z);
        }
        if (opt == 2) {
            int x = read(), y = read();
            // if(T.t[x].mp>T.t[y].mp)x^=y^=x^=y;
            printf("%d\n", QUERY(x, y));
        }
        if (opt == 3) {
            int x = read(), z = read();
            TT.Update(1, 1, n, T.t[x].mp, T.t[x].mp + T.t[x].siz - 1, z);
        }
        if (opt == 4) {
            int x = read();
            printf("%d\n", TT.query(1, 1, n, T.t[x].mp, T.t[x].mp + T.t[x].siz - 1));
        }
    }
    return 0;
}

你可能感兴趣的:(砍树操作——树链剖分,考验美术?——图论,有趣段子——线段树/树状数组)