首页 > 其他分享 >ZJOI2008, 树的统计 树链剖分模板

ZJOI2008, 树的统计 树链剖分模板

时间:2022-12-30 23:22:24浏览次数:46  
标签:val 剖分 int hs top mid 树链 ZJOI2008 id

//题意:给定一棵树,现在我需要询问以下操作
//      1.q,u之间的最小值
//      2.q,u之间的简单路径的权值和
//      3.修改树上q点的权值
//思路:如果是在一段序列上的问题,我们可以直接线段树解决,但是这是一棵树,我们也无法将两点之间的简单路径转化为一段连续区间
//      所以我们使用树链剖分(重链剖分),如此一来我们可以用类似于倍增的方式维护出一条简单路径(简单路径上的信息可以用线段树维护)
//        因为我们重链剖分后,按照优先遍历重儿子的理念,一条重链一定在dfs序中是连续的,这也方便了我们用线段树查询
// 
//复杂度:每次维护区间信息 区间剖分logn*线段树维护*logn 所以是O(logn*logn)
//   
#include<bits/stdc++.h>
using namespace std;
#define int long long

const int N = 3 * 1e5;

int n, m, a[N];
vector<int> e[N];
int l[N], r[N], idx[N];
int sz[N], hs[N], tot, top[N], dep[N], fa[N];

struct info {
    int maxv, sum;//要记录的路径最大值,与路径之和
    info(int a=0, int b=0) {
        maxv = a, sum = b;
    }
};

info operator +(const info& l, const info& r) {
    return { max(l.maxv, r.maxv), l.sum + r.sum };//重载合并运算
}

struct node {
    info val;
}seg[N * 4];

void update(int id) {
    seg[id].val = seg[id * 2].val + seg[2 * id + 1].val;
}

void build(int id, int l, int r) {
    if (l == r) {
        //我们是对dfs序建树,所以这里要注意是dfs序中l号点
        seg[id].val = { a[idx[l]],a[idx[l]] };
    }
    else {
        int mid = (l + r) / 2;
        build(id * 2, l, mid);
        build(id * 2 + 1, mid + 1, r);
        update(id);
    }
}

void change(int id, int l, int r, int pos, int val) {//本题只有单点修改
    if (l == r) {
        seg[id].val = { val,val };
    }
    else {
        int mid = (l + r) / 2;
        if (pos <= mid) change(id * 2, l, mid, pos, val);
        else change(id * 2 + 1, mid + 1, r, pos, val);
        update(id);
    }
}

info query(int id, int l, int r, int ql, int qr) {
    if (l == ql && r == qr) return seg[id].val;
    int mid = (l + r) / 2;
    if (qr <= mid) return query(id * 2, l, mid, ql, qr);
    else if (ql > mid) return query(id * 2 + 1, mid + 1, r, ql, qr);
    else {
        return query(id * 2, l, mid, ql, mid) +
            query(id * 2 + 1, mid + 1, r, mid + 1, qr);
    }
}

void dfs1(int u, int f) {//第一遍dfs,求子树大小,重儿子,父亲,深度
    sz[u] = 1;
    hs[u] = -1;
    fa[u] = f;
    dep[u] = dep[f] + 1;
    for (auto v : e[u]) {
        if (v == f) continue;
        dfs1(v, u);
        sz[u] += sz[v];
        if (hs[u] == -1 || sz[v] > sz[hs[u]])
            hs[u] = v;
    }
}

void dfs2(int u, int t) {//第二次dfs,求每个点的dfs序,重链上的链头元素
    top[u] = t;
    l[u] = ++tot;
    idx[tot] = u;
    if (hs[u] != -1) {
        dfs2(hs[u], t);
    }//优先遍历重链,如此一来将重链的dfs序搞成连续的
    for (auto v : e[u]) {
        if (v != fa[u] && v != hs[u])
            dfs2(v, v);//这里很重要,最开始被dls坑了
    }
    r[u] = tot;
}

info query1(int u, int v) {
    info ans{ (int)-1e9,0 };
    while (top[u] != top[v]) {
        if (dep[top[u]] < dep[top[v]]) {
            ans = ans + query(1, 1, n, l[top[v]], l[v]);
            v = fa[top[v]];
        }
        else {
            ans = ans + query(1, 1, n, l[top[u]], l[u]);
            u = fa[top[u]];
        }
    } 
    if (dep[u] <= dep[v]) ans = ans + query(1, 1, n, l[u], l[v]);
    else ans = ans + query(1, 1, n, l[v], l[u]);
    return ans;
}

signed main() {
    scanf("%d", &n);
    for (int i = 1; i < n; i++) {
        int u, v;
        scanf("%lld%lld", &u, &v);
        e[u].push_back(v);
        e[v].push_back(u);
    }
    for (int i = 1; i <= n; i++) scanf("%lld", &a[i]);
    dfs1(1, 0);
    dfs2(1, 1);
    build(1, 1, n);
    scanf("%lld", &m);
    for (int i = 0; i < m; i++) {
        int u, v;
        static char op[10];
        scanf("%s%lld%lld", op, &u, &v);
        if (op[0] == 'C') {
            change(1, 1, n, l[u], v);
        }
        else {
            info ans = query1(u, v);
            if (op[1] == 'M') printf("%d\n", ans.maxv);
            else printf("%d\n", ans.sum);
        }
    }
    return 0;
}



/*19
1 2
2 3
3 4
4 5
5 6
5 7
3 10
2 8
8 9
1 11
11 12
12 13
12 14
1 15
15 16
16 17
17 18
16 19*/

 

标签:val,剖分,int,hs,top,mid,树链,ZJOI2008,id
From: https://www.cnblogs.com/Aacaod/p/17016038.html

相关文章

  • 处理手法学习(树链剖分)
    今天vp了两场cf,感觉对开眼界还是很有用的,手感也回来了点首先给出一些点,如何找出是否属于同一条链首先暴力方法就是每次dfs,在分叉大于2的地方看看是否包含所有的点这是个......
  • BZOJ 3252 攻略(长链剖分模板 链的常用性质 贪心)
    BZOJ3252攻略​ 给定一棵带边权的树,选择k个叶子结点,使这些叶子结点与根节点相连,形成k条路径。输出被路径覆盖的所有边的边权和的最大值(同一条边若被重复覆盖只算一......
  • 最大面积最小的三角剖分 Minimax Triangulation
    #include<iostream>#include<algorithm>#include<cmath>#include<vector>usingnamespacestd;#definemistake(P)(fabs(P)<eps?0:(P<0?-1:1))......
  • 边权树链剖分
    边权树链剖分一般的树链剖分都是维护的点权,而如果要维护边权怎么办呢?思路我们可以把边权转换为点权。一个点有很多个儿子,但只有一个父亲。如果一个节点的点权记录到其儿......
  • 重链剖分学习笔记
    最近在机房自习复习了一下这些东西,主要给自己看的。参考文献:OIWiki《算法竞赛》(罗勇军,郭卫斌著)1重链剖分简介1.1重链剖分基本性质树链剖分,就是把树分成若......
  • 重链剖分学习笔记
    最近在机房自习复习了一下这些东西,主要给自己看的。参考文献:OIWiki《算法竞赛》(罗勇军,郭卫斌著)1重链剖分简介1.1重链剖分基本性质树链剖分,就是把树分成若......
  • 树链剖分学习笔记
    树链剖分学习笔记简介树链剖分是一种可以把树丢到线段树上维护的一种算法,时间复杂度为\(O(n\log^2n)\)。思路一、一些概念1.重儿子:如果一个点有儿子,那么所有儿子中......
  • P3128 [USACO15DEC]Max Flow P(树上倍增和树链剖分)
    思路1(树上倍增$+$树上差分)每次都修改一条从\(u\)到\(v\),不就是树上差分的专门操作吗??直接用倍增求\(LCA\),每次\(d[u]++,d[v]++,d[LCA(u,v)]--,d[f[LCA(u,v)][0]]--\)。......
  • 浅谈树链剖分
    树链剖分定理重儿子:一个节点所有儿子中,子树大小最大的儿子即为重儿子,如有多个,任取一个即可。轻儿子:除了重儿子外的所有儿子。重边:父节点\(\to\)重儿子的边。重链:由......
  • 树链剖分
    树链剖分0x00绪言在阅读这篇文章前,确保你已学会你下内容:线段树深度优先遍历会了这些就可以开始阅读本篇文章了。0x01什么是树剖把一棵树拆成若干个不相交的链,然......