[NOIp2016]天天爱跑步 线段树合并

[NOIp2016]天天爱跑步

LG传送门

作为一道被毒瘤出题人们玩坏了的NOIp经典题,我们先不看毒瘤的“动态爱跑步”和“天天爱仙人掌”,回归一下本来的味道。

对于一个人,他的路程会分为两段,一段向上(根),一段向下,考虑在向上过程中他能产生贡献的观察者具有什么性质:设出发点深度为\(dep[x]\),观察者深度为\(dep[y]\),观察的时间为\(t\),需满足\(dep[x] - dep[y] = t\),换句话说就是\(dep[y] + t = dep[x]\)。向下那一段的推导类似,下面的部分也只以向上的路径为例来解释。

现在我们记录每个点下方出发点深度为\(dep[x]\)的人数,需要对于每个出发点更新出发点到出发点与目的地的lca的所有点,想到到开一颗权值线段树,用线段树合并不断把信息往上传,在lca处消除这次更新的影响,这样一来直接我们就可以在树上统计答案了。向下路径的处理与向上路径类似,这里就不再赘述,上代码。

#include <cstdio>
#include <cctype>
#include <vector>
#define R register
#define I inline
#define B 1000000
using namespace std;
const int N = 300001, M = 15000007;
char buf[B], *p1, *p2;
I char gc() { return p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, B, stdin), p1 == p2) ? EOF : *p1++; }
I int rd() {
    R int f = 0;
    R char c = gc();
    while (c < 48 || c > 57)
        c = gc();
    while (c > 47 && c < 58)
        f = f * 10 + (c ^ 48), c = gc();
    return f;
}
int w[N], s[N], t[N], p[N], q[N], d[N], r[N], u[N], v[N], o[N], n, C, S;
struct T{int f, p, q;}e[M];
vector <int> g[N];
I void swp(int &x, int &y) { x ^= y, y ^= x, x ^= y; }
void dfs1(int x, int f) {
    p[x] = f, d[x] = d[f] + 1, t[x] = 1;
    for (R int i = 0, y, m = 0; i < s[x]; ++i)
        if ((y = g[x][i]) ^ f) {
            dfs1(y, x), t[x] += t[y];
            if (t[y] > m)
                m = t[y], q[x] = y;
        }
}
void dfs2(int x, int t) {
    r[x] = t;
    if (q[x])
        dfs2(q[x], t);
    for (R int i = 0, y; i < s[x]; ++i)
        if ((y = g[x][i]) ^ p[x] && y ^ q[x])
            dfs2(y, y);
}
I int lca(int x, int y) {
    while (r[x] ^ r[y]) {
        if (d[r[x]] < d[r[y]])
            swp(x, y);
        x = p[r[x]];
    }
    if (d[x] > d[y])
        swp(x, y);
    return x;
}
void ins(int &k, int l, int r, int x, int v) {
    if (!k)
        k = ++C;
    e[k].f += v;
    if (l == r)
        return ;
    R int m = l + r >> 1;
    if (x <= m)
        ins(e[k].p, l, m, x, v);
    else
        ins(e[k].q, m + 1, r, x, v);
}
int mrg(int k, int t) {
    if (!k)
        return t;
    if (!t)
        return k;
    e[k].f += e[t].f, e[k].p = mrg(e[k].p, e[t].p), e[k].q = mrg(e[k].q, e[t].q);
    return k;
}
int qry(int k, int l, int r, int x) {
    if (l == r)
        return e[k].f;
    R int m = l + r >> 1;
    if (x <= m)
        return qry(e[k].p, l, m, x);
    else
        return qry(e[k].q, m + 1, r, x);
}
void dfs(int x) {
    for (R int i = 0, y; i < s[x]; ++i)
        if ((y = g[x][i]) ^ p[x])
            dfs(y), u[x] = mrg(u[x], u[y]), v[x] = mrg(v[x], v[y]);
    o[x] = qry(u[x], 1, S, w[x] + d[x]) + qry(v[x], 1, S, w[x] - d[x] + n);
}
int main() {
    R int m, i, x, y, a;
    n = rd(), m = rd(), S = n << 1;
    for (i = 1; i < n; ++i)
        x = rd(), y = rd(), g[x].push_back(y), g[y].push_back(x);
    for (i = 1; i <= n; ++i)
        s[i] = g[i].size(), w[i] = rd();
    dfs1(1, 0), dfs2(1, 1);
    for (i = 1; i <= m; ++i)
        x = rd(), y = rd(), a = lca(x, y), ins(u[x], 1, S, d[x], 1), ins(u[a], 1, S, d[x], -1), ins(v[y], 1, S, d[x] - (d[a] << 1) + n, 1), ins(v[p[a]], 1, S, d[x] - (d[a] << 1) + n, -1);
    dfs(1);
    for (i = 1; i <= n; ++i)
        printf("%d ", o[i]);
    return 0;
}

因为减的结果可能有负数,所以加上一个\(n\),值域变两倍。

转载于:https://www.cnblogs.com/cj-chd/p/10353138.html

Stkcd [股票代码] ShortName [股票简称] Accper [统计截止日期] Typrep [报表类型编码] Indcd [行业代码] Indnme [行业名称] Source [公告来源] F060101B [净利润现金净含量] F060101C [净利润现金净含量TTM] F060201B [营业收入现金含量] F060201C [营业收入现金含量TTM] F060301B [营业收入现金净含量] F060301C [营业收入现金净含量TTM] F060401B [营业利润现金净含量] F060401C [营业利润现金净含量TTM] F060901B [筹资活动债权人现金净流量] F060901C [筹资活动债权人现金净流量TTM] F061001B [筹资活动股东现金净流量] F061001C [筹资活动股东现金净流量TTM] F061201B [折旧摊销] F061201C [折旧摊销TTM] F061301B [公司现金流1] F061302B [公司现金流2] F061301C [公司现金流TTM1] F061302C [公司现金流TTM2] F061401B [股权现金流1] F061402B [股权现金流2] F061401C [股权现金流TTM1] F061402C [股权现金流TTM2] F061501B [公司自由现金流(原有)] F061601B [股权自由现金流(原有)] F061701B [全部现金回收率] F061801B [营运指数] F061901B [资本支出与折旧摊销比] F062001B [现金适合比率] F062101B [现金再投资比率] F062201B [现金满足投资比率] F062301B [股权自由现金流] F062401B [企业自由现金流] Indcd1 [行业代码1] Indnme1 [行业名称1] 季度数据,所有沪深北上市公司的 分别包含excel、dta数据文件格式及其说明,便于不同软件工具对数据的分析应用 数据来源:基于上市公司年报及公告数据整理,或相关证券交易所、各部委、省、市数据 数据范围:基于沪深北证上市公司 A股(主板、中小企业板、创业板、科创板等)数据整理计算
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值