zoukankan      html  css  js  c++  java
  • 【LOJ】#3014. 「JOI 2019 Final」独特的城市(长链剖分)

    LOJ#3014. 「JOI 2019 Final」独特的城市(长链剖分)

    显然我们画一条直径,容易发现被统计的只可能是直径某个距离较远的端点到这个点的路径上的值

    用一个栈统计可以被统计的点,然后我们把这棵树长链剖分,每次在所有轻儿子中找深度最大的,去掉距离u小于这个深度的栈里的点,然后去计算u的重儿子

    然后去掉距离u小于重儿子深度栈里的点,但是要再把u加进去,再遍历u的其他儿子

    最后重新去掉u,计算答案,用直径两端当根都做一遍,取深度较大的那个

    统计的话直接在外面开一个数组,弹出弹入的时候判断以下就好了

    #include <bits/stdc++.h>
    #define fi first
    #define se second
    #define pii pair<int,int>
    #define mp make_pair
    #define pb push_back
    #define space putchar(' ')
    #define enter putchar('
    ')
    #define eps 1e-10
    #define MAXN 200005
    #define ba 47
    //#define ivorysi
    using namespace std;
    typedef long long int64;
    typedef unsigned int u32;
    typedef double db;
    template<class T>
    void read(T &res) {
        res = 0;T f = 1;char c = getchar();
        while(c < '0' || c > '9') {
    	if(c == '-') f = -1;
    	c = getchar();
        }
        while(c >= '0' && c <= '9') {
    	res = res * 10 +c - '0';
    	c = getchar();
        }
        res *= f;
    }
    template<class T>
    void out(T x) {
        if(x < 0) {x = -x;putchar('-');}
        if(x >= 10) {
    	out(x / 10);
        }
        putchar('0' + x % 10);
    }
    struct node {
        int to,next;
    }E[MAXN * 2];
    int N,M,dep[MAXN],mx[MAXN],son[MAXN],head[MAXN],sumE,S,T,c[MAXN];
    int sta[MAXN],top,all;
    int cnt[MAXN],ans[MAXN];
    void add(int u,int v) {
        E[++sumE].to = v;
        E[sumE].next = head[u];
        head[u] = sumE;
    }
    void dfs(int u,int fa) {
        son[u] = 0;
        mx[u] = dep[u];
        for(int i = head[u] ; i ; i = E[i].next) {
    	int v = E[i].to;
    	if(v != fa) {
    	    dep[v] = dep[u] + 1;
    	    dfs(v,u);
    	    if(mx[v] > mx[son[u]]) son[u] = v;
    	    mx[u] = max(mx[u],mx[v]);
    	}
        }
    }
    void ins(int x) {
        sta[++top] = x;
        x = c[x];
        if(cnt[x] == 0) ++all;
        ++cnt[x];
    }
    void del() {
        int x = sta[top--];
        x = c[x];
        if(cnt[x] == 1) --all;
        --cnt[x];
    }
    void calc(int u,int fa) {
        if(!son[u]) {
    	ans[u] = max(ans[u],all);return;
        }
        int ol = 0;
        for(int i = head[u] ; i ; i = E[i].next) {
    	int v = E[i].to;
    	if(v != fa && v != son[u]) {
    	    ol = max(ol,mx[v] - dep[u]);
    	}
        }
        while(top && dep[sta[top]] + ol >= dep[u]) del();
        ins(u);calc(son[u],u);
        while(top && dep[sta[top]] + mx[son[u]] - dep[u] >= dep[u]) del();
        for(int i = head[u] ; i ; i = E[i].next) {
    	int v = E[i].to;
    	if(v != fa && v != son[u]) {
    	    if(sta[top] != u) ins(u);
    	    calc(v,u);
    	}
        }
        while(top && dep[sta[top]] + mx[son[u]] - dep[u] >= dep[u]) del();
        ans[u] = max(ans[u],all);return;
    }
    void Solve() {
        read(N);read(M);
        int a,b;
        for(int i = 1 ; i < N ; ++i) {
    	read(a);read(b);
    	add(a,b);add(b,a);
        }
        for(int i = 1 ; i <= N ; ++i) read(c[i]);
        dep[1] = 0;dfs(1,0);
        S = 1;
        for(int i = 2 ; i <= N ; ++i) {
    	if(dep[i] > dep[S]) S = i;
        }
        dep[S] = 0;dfs(S,0);
        T = 1;
        for(int i = 2 ; i <= N ; ++i) {
    	if(dep[i] > dep[T]) T = i;
        }
        dep[S] = 0;dfs(S,0);top = 0;
        calc(S,0);
        memset(cnt,0,sizeof(cnt));all = 0;
        dep[T] = 0;dfs(T,0);top = 0;
        calc(T,0);
        for(int i = 1 ; i <= N ; ++i) {
    	out(ans[i]);enter;
        }
    }
    int main() {
    #ifdef ivorysi
        freopen("f1.in","r",stdin);
    #endif
        Solve();
    }
    
    
  • 相关阅读:
    [BJOI2019] 光线
    [BJOI2019]奥术神杖
    [HNOI2014]江南乐
    [SDOI2018]荣誉称号
    [APIO2015]雅加达的摩天楼
    [TJOI2015]线性代数
    【CF163E 】e-Government
    【CF917D】Stranger Trees
    网络流(四)dinic算法
    网络流(三)最大流最小割定理
  • 原文地址:https://www.cnblogs.com/ivorysi/p/10943426.html
Copyright © 2011-2022 走看看