Codeforces Round 907 (Div. 2) F. A Growing Tree

传送门

写在前边

这道题几乎是牛客上的原题,只不过修改了最后的询问每个节点的值,而是将询问节点值转为了3操作。解题思路是完全一致的

华华和月月种树

题意

给定一个初始的根节点,权值为0。有两种操作
1. 在节点v上添加一个编号为 sz+1的子顶点,其中 sz是树的当前大小。新顶点的数值为 0
2. 在顶点 v的子树中所有顶点的数值中添加 x

题目有很多种解法,比如(差分树状数组,线段树,树上前缀和等等)

因为有区间修改和单点查询,所以使用了线段树

思路:
首先,我们需要离线的处理输入,我们可以遍历每次的输入,然后建树。并且需要对每个节点赋予一个时间戳。因为在我们处理某个节点及其子树的权值时,我们希望它是一段连续的区间,这样会比较好处理。对于2操作,我们就直接进行区间修改。而对于1操作,我们只需要把这个节点的权值赋为0即可。所以我们先根据输入建树,然后跑一遍DFS给每个节点一个时间戳,并计算出以每个节点为根的子树大小

#include <bits/stdc++.h>

using i64 = long long;
using PII = std::pair<i64,i64>;
#define int i64
#define yes std::cout << "YES\n";
#define no std::cout << "NO\n";

struct SegmentTree {
    const int n;
    struct Node {
        int l, r;
        i64 sum, add;
    };
    std::vector<Node> tr;
    SegmentTree(int n) : n(n), tr(4 << std::__lg(n)) {
        std::function<void(int, int, int)> build = [&](int u, int l, int r) {
            tr[u].l = l, tr[u].r = r;
            if(l == r) tr[u].sum = 0;
            else {
                int mid = l + r >> 1;
                build(u << 1, l, mid); build(u << 1 | 1, mid + 1, r);
                pushup(u);
            }
        };
        build(1, 1, n);
    }
    SegmentTree(std::vector<i64> init) : SegmentTree(init.size() - 1) {
        
    }
    void pushup(int u) {
        tr[u].sum = tr[u << 1].sum + tr[u << 1 | 1].sum; 
    }
    void pushdown(int u) {
        if(tr[u].add) {
            tr[u << 1].sum += (tr[u << 1].r - tr[u << 1].l + 1) * tr[u].add;
            tr[u << 1 | 1].sum += (tr[u << 1 | 1].r - tr[u << 1 | 1].l + 1) * tr[u].add;
            tr[u << 1].add += tr[u].add;
            tr[u << 1 | 1].add += tr[u].add;
            tr[u].add = 0;
        }
    }
    void modify(int u, int l, int r, i64 k) {
        if(tr[u].l >= l && tr[u].r <= r){
            tr[u].sum += (tr[u].r - tr[u].l + 1) * k;
            tr[u].add += k;
        } else {
            pushdown(u);
            int mid = tr[u].l + tr[u].r >> 1;
            if(l <= mid) modify(u << 1, l, r, k);
            if(r > mid) modify(u << 1 | 1, l, r, k);
            pushup(u);
        }
    }
    i64 query(int u, int l, int r) {
        if(tr[u].l > r || tr[u].r < l) return 0;
        if(tr[u].l >= l && tr[u].r <= r) return tr[u].sum;
        pushdown(u);
        int mid = tr[u].l + tr[u].r >> 1;
        return query(u << 1, l, r) + query(u << 1 | 1, l, r);
    }
};
//SegmentTree seg(w);  


void solve() {
    int n;
    std::cin >> n;

    int c = 2; //下一个节点的编号
    std::vector<std::array<int,3>> a(n);
    std::vector<std::vector<int>> adj(n + 10);
    for (int i = 0; i < n; i ++) {
        int op, x, y;
        std::cin >> op;
        if (op == 1) {
            std::cin >> x;
            a[i] = {op,x,-1};
            adj[x].push_back(c);
            c ++;
        } else {
            std::cin >> x >> y;
            a[i] = {op,x,y};
        }
    }
    std::vector<int> d(n + 10);//每个节点的时间戳
    std::vector<int> sz(n + 10);//每个点为根的子树大小,包括自己

    int t = 1;
    auto dfs = [&](auto&& dfs,int u) -> int {
        sz[t] = 1;
        d[u] = t;
        t ++;
        int ans = 0;
        for (auto v : adj[u]) {
            ans += dfs(dfs,v);
        }
        sz[d[u]] += ans;
        return sz[d[u]];
    };

    dfs(dfs,1);

    SegmentTree seg(n + 1);

    int k = c;//最大节点,此节点还未被创建

    c = 2;
    for (int i = 0; i < n; i ++) {
        auto [op,x,y] = a[i];
        if (op == 1) {//要把当前节点的值变为0.为了方便,可以先查一次,再加它的相反数
            int l = d[c];
            int va = seg.query(1,l,l);
            seg.modify(1,l,l,-va);
            c ++;
        } else {//节点的时间戳为左区间,时间戳+子树大小 - 1即为右区间
            int l = d[x],r = d[x] + sz[d[x]] - 1;
            seg.modify(1,l,r,y);
            
        }
    }
    for (int i = 1; i <= k - 1; i ++) {
        std::cout << seg.query(1,d[i],d[i]) << " \n"[i == k - 1];
    }

}       

signed main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(nullptr); 
    
    int T = 1;
    
    std::cin >> T;

    while (T -- ) {
        solve();
    }
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值