zoukankan      html  css  js  c++  java
  • AGC018D Tree and Hamilton Path

    Link
    先求出最长的Hamilton回路。
    对于原树中的一条边((u,v,w)),不难发现它的贡献上界是(min(size_u,size_v)*w)。(这里的(size)指的是把这条边断掉之后两个子树的大小)
    事实上存在一种方案使得每条边的贡献上界都取到,这个方案中相邻两个点在路上的路径都会经过树的重心。
    然后我们考虑去掉一条最短的可能在最优解中的路径得到最长的Hamilton路径,显然这条路径就是树的重心连出去的最小的一条边(如果重心是一条边的话,那么这条路径就是树的重心)。

    #include<cstdio>
    #include<vector>
    #include<utility>
    using i64=long long;
    using pi=std::pair<int,int>;
    const int N=100007;
    std::vector<pi>e[N];int n,root,mn,size[N],ef[N],es[N];i64 ans;
    int read(){int x;scanf("%d",&x);return x;}
    void dfs(int u,int fa)
    {
        int mx=0,v;size[u]=1,ef[u]=es[u]=1e9;
        for(pi x:e[u]) if((v=x.first)^fa) dfs(v,u),size[u]+=size[v],mx=std::max(mx,size[v]),es[u]=std::min(es[u],x.second),ans+=2ll*x.second*std::min(size[v],n-size[v]); else ef[u]=x.second;
        mx=std::max(mx,n-size[u]),mx<mn? mn=mx,root=u:0;
    }
    int main()
    {
        mn=n=read();
        for(int i=1,u,v,w;i<n;++i) u=read(),v=read(),w=read(),e[u].push_back({v,w}),e[v].push_back({u,w});
        dfs(1,0),ans-=(mn*2==n? ef[root]:std::min(ef[root],es[root]));
        printf("%lld",ans);
    }
    
  • 相关阅读:
    javascript小测试
    js设计模式--策略模式
    js设计模式--迭代器模式
    面试题自我解析
    js设计模式--工厂模式
    js设计模式--单体模式
    Python学习一:Python简介
    Angularjs学习笔记《一》
    实现字体最小,不是默认的12px,可以在视觉上最小达到6px
    css3 transform之后,图片的清晰度下降解决方式
  • 原文地址:https://www.cnblogs.com/cjoierShiina-Mashiro/p/12694220.html
Copyright © 2011-2022 走看看