zoukankan      html  css  js  c++  java
  • 仙人掌图直径

    题:https://www.luogu.com.cn/problem/P4244

    分析:定义f[i]表示dfs过程中 i 节点到子树叶子节点的最大距离;

       考虑俩种边的俩种情况,【1】假设枚举边为树边(桥),那么对答案的更新则为f[u]+f[v]+1,表示在枚举v之前的最大叶子节点距离和当前v节点到叶子节点的最大距离,同时用f[v]更新f[u].

                  【2】考虑环,环对答案的更新无非就选取俩个点i,j 的f[i]+f[j]+abs(i-j)。那么对于当前的i,只要找到最大的f[j]-j ,所以维护这样一个环的单调队列。

    #include<bits/stdc++.h>
    using namespace std;
    #define pb push_back
    #define ull unsigned long long
    #define pii pair<int,int>
    #define MP make_pair
    typedef long long ll;
    const ll INF=1e18;
    const int M=5e5+6;
    int deep[M],a[M],fa[M],f[M],que[M],low[M],dfn[M];
    vector<int>g[M];
    int ans,cnt;
    void solve(int u,int v){
        int sum=deep[v]-deep[u]+1;
        for(int i=v;i!=u;i=fa[i])
            a[sum--]=f[i];
        a[1]=f[u];
        sum=deep[v]-deep[u]+1;
        que[1]=1;
        int head=1,tail=1;
        for(int i=1;i<=sum;i++)
            a[i+sum]=a[i];
        for(int i=2;i<=sum+(sum>>1);i++){
            if(i-que[head]>(sum>>1))head++;///因为俩点间的距离是最短距离,所以把选的范围缩小为环的一半,这也是为什么要枚举3/2环
            ans=max(ans,a[i]+a[que[head]]+i-que[head]);///对于更新答案,a[i]+i是确定的,那么只要找到最大的a[j]-j即可,也就是维护a[j]-j的单调队列,保证头部是最大的
            while(head<=tail&&a[i]-i>=a[que[tail]]-que[tail])
                tail--;
            que[++tail]=i;
        }
        for(int i=2;i<=sum;i++)
            f[u]=max(f[u],a[i]+min(i-1,sum-i+1));///因为定义俩点最短距离为2,所以选取环的较短边。
    }
    void dfs(int u){
        dfn[u]=low[u]=++cnt;
        for(auto v:g[u]){
            if(v!=fa[u]){
                if(!dfn[v]){
                    deep[v]=deep[u]+1;
                    fa[v]=u;
                    dfs(v);
                }
                low[u]=min(low[u],low[v]);
                if(low[v]>dfn[u]){
                    ans=max(ans,f[u]+f[v]+1);
                    f[u]=max(f[u],f[v]+1);
                }
            }
        }
        for(auto v:g[u]){
            if(fa[v]!=u&&dfn[v]>dfn[u])
                solve(u,v);
        }
    }
    int main(){
        int n,m;
        scanf("%d%d",&n,&m);
        while(m--){
            int k,u=0,v;
            scanf("%d",&k);
            while(k--){
                scanf("%d",&v);
                if(u)g[u].pb(v),g[v].pb(u);
                u=v;
            }
        }
        dfs(1);
        printf("%d
    ",ans);
        return 0;
    }
    View Code
  • 相关阅读:
    Windows平板优化设置
    MAC OS UI设计
    使用bat/vbs/ahk对Windows下进行自动化操作
    C#在高性能计算领域为什么性能却如此不尽人意
    自定义多重搜索
    CF797E Array Queries
    标记永久化学习笔记
    P7200 [COCI2019-2020#1] Lutrija
    P1075 [NOIP2012 普及组] 质因数分解
    基础数论重学笔记
  • 原文地址:https://www.cnblogs.com/starve/p/13533369.html
Copyright © 2011-2022 走看看