考虑把树展开,单独把叶子节点拿出来

于是可以把控制点\(x\)的,抽象成是在它叶子节点间连权值为\(c_x\)的边

显然只用在\(x\)子树的最左边的叶子节点和最右边的叶子节点的下一个节点连边(最后一个叶子节点的下一个节点为 \(n+1\)),跑最小生成树即可

正确性证明的话,设叶子节点的权值分别为\(x_1,x_2……x_n\),做差分\(y_i=x_{i+1}-x_i\),显然\(\sum \limits _{i=1}^n y_i =0\)

正确性的话,感性理解一下吧QAQ,有理有据的感性理解

代码:

//OwO
#include <bits/stdc++.h>
#define N 200010
#define rep(i,x,y) for(i=x;i<=y;++i)
#define add(x,y) g[x].push_back(y)
#define ll long long
#define pii pair<int,int>
using namespace std;

ll c[N];
int dfn[N],sz[N],cnt,tot,fa[N],qwq[N],l[N],r[N],rr[N],n,lf=0;
vector<int> g[N];
struct ed{
    int u,v,p;
    ll w;
    bool operator <(const ed &x)const{ return w<x.w; }
}e[N];
void ADD(int u,int v,int p,ll w){ e[++cnt]=(ed){u,v,p,w}; }

pii dfs(int x,int fa){
    pii lst;
    int pl=0;
    if(g[x].size()<=1 && x!=1) l[x]=r[x]=++lf;
    for(int i=0;i<g[x].size();++i){
        int to=g[x][i];
        if(to!=fa){
            pii tmp=dfs(to,x);
            l[x]=min(l[x],tmp.first);
            r[x]=max(r[x],tmp.second);
        }
    }
    return make_pair(l[x],r[x]);
}

int find(int x){ return (fa[x]==x)?x:fa[x]=find(fa[x]); }

ll solve1(){
    int i;ll ans=0;
    rep(i,1,cnt){
        int u=find(e[i].u),v=find(e[i].v);
        if(u==v) continue;
        if(u>v) swap(u,v);
        fa[v]=u;
        ans+=e[i].w;
    }
    return ans;
}

void solve2(){
    int i,nxt,j;tot=0;
    rep(i,1,n+1) fa[i]=i;
    for(i=1;i<=cnt;i=nxt){
        for(j=i;e[j].w==e[i].w;++j){
            int u=find(e[j].u),v=find(e[j].v);
            if(u!=v) qwq[++tot]=e[j].p;
        }
        nxt=j;
        for(j=i;e[j].w==e[i].w;++j){
            int u=find(e[j].u),v=find(e[j].v);
            if(u>v) swap(u,v);
            if(u!=v) fa[v]=u;
        }
    }
}

int main(){
    int i,x,y;
    scanf("%d",&n);
    rep(i,1,n) scanf("%I64d",&c[i]),fa[i]=i;fa[n+1]=n+1;
    rep(i,1,n-1){
        scanf("%d%d",&x,&y);
        add(x,y),add(y,x);
    }
    memset(l,0x3f,sizeof(l));
    memset(r,0,sizeof(r));
    dfs(1,0);rr[1]=n+1;
    rep(i,1,n) ADD(l[i],r[i]+1,i,c[i]);
    sort(e+1,e+cnt+1);
    cout<<solve1()<<" ";
    solve2();
    sort(qwq+1,qwq+tot+1);
    cout<<tot<<endl;
    rep(i,1,tot) printf("%d ",qwq[i]);
}

CF1120D(神奇的构造+最小生成树)的更多相关文章

  1. Kruskal算法构造最小生成树

    Kruskal算法来构造最小生成树,我总结了分为以下步骤: (1)建图,构造Kruskal边集,边集元素应该包括该边的起始顶点.终止顶点.权值: (2)将边集按权值从小到大的顺序进行排序: (3)从小 ...

  2. c/c++ 用克鲁斯卡尔(kruskal)算法构造最小生成树

    c/c++ 用克鲁斯卡尔(kruskal)算法构造最小生成树 最小生成树(Minimum Cost Spanning Tree)的概念: 假设要在n个城市之间建立公路,则连通n个城市只需要n-1条线路 ...

  3. c/c++ 用普利姆(prim)算法构造最小生成树

    c/c++ 用普利姆(prim)算法构造最小生成树 最小生成树(Minimum Cost Spanning Tree)的概念: ​ 假设要在n个城市之间建立公路,则连通n个城市只需要n-1条线路.这时 ...

  4. 图的建立(邻接矩阵)+深度优先遍历+广度优先遍历+Prim算法构造最小生成树(Java语言描述)

    主要参考资料:数据结构(C语言版)严蔚敏   ,http://blog.chinaunix.net/uid-25324849-id-2182922.html   代码测试通过. package 图的建 ...

  5. MST性质(用于构造最小生成树)

    描述:假设N=(V,{E})是一个连通网,U是顶点集V的一个非空子集.若(u,v)是一条具有最小权值(代价)的边,其中u∈U,v∈V-U,则必存在一棵包含边(u,v)的最小生成树. 证明: 假设网N的 ...

  6. 数据结构-图-Java实现:有向图 图存储(邻接矩阵),最小生成树,广度深度遍历,图的连通性,最短路径1

    import java.util.ArrayList; import java.util.List; // 模块E public class AdjMatrixGraph<E> { pro ...

  7. CSU 1116 Kingdoms(枚举最小生成树)

    题目链接:http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1116 解题报告:一个国家有n个城市,有m条路可以修,修每条路要一定的金币,现在这个国家只 ...

  8. poj1251 最小生成树

    Description The Head Elder of the tropical island of Lagrishan has a problem. A burst of foreign aid ...

  9. ACM: meixiuxiu学图论-并查集-最小生成树-解题报告

    /* 最小生成树,最小环的最大权值按照排序后去构建最小生成树就可以了,注意遇到的第一个根相同的点就记录权值,跳出,生成的环就是最小权值环. */ //AC代码: #include"iostr ...

随机推荐

  1. 推荐一款关于MongoDB日志分析的工具--Mtools

    一. 需求背景 MongoDB数据库的强大的文档模型使其成为处理数据的最佳方式.文档适用于广泛的流行数据模型,支持各种各样的场景.文档模型可以包含键值.关系数据集和图形数据集,当然,还可以包含父子关系 ...

  2. Linux内核的冷热缓存

    缓存为什么会有冷热? 究其原因,是因为对于内存的访问,可能是CPU发起的,也可以是DMA设备发起的. 如果是CPU发起的,在CPU的硬件缓存中,就会保存相应的页内容.如果这个页本来没有存在于硬件缓存中 ...

  3. 统计numpy数组中最频繁出现的值

    arr = np.array([[1,2,100,4,5,6],[1,1,100,3,5,5],[2,2,4,4,6,6]]) 方法一: count = np.bincount(arr[:,2]) # ...

  4. idea软件破解汉化

    →http://idea.lanyus.com/上可以找到最新的破解补丁,下载并放到软件的bin目录下  →更改bin目录下的两个文件:Idea.exe.vmoptions和Idea64.exe.vm ...

  5. 面向对象_del

    老师的博客http://www.cnblogs.com/Eva-J/articles/7351812.html#_label7 内置的方法有很多不一定全都在object中 #python3中,所有类都 ...

  6. linux -- 添加、修改、删除路由

    在日常的使用中,或者在服务器中,有两个网卡配置两个地址,访问不同的网络段,这种情况是非常常见的现象,但是,我们需要额外的添加路由表来决定发送的数据包经过正确的网关和interface才能正确的进行通信 ...

  7. JavaScript 函数闭包的应用

    一.模仿块级作用域 JavaScript 没有块级作用域的概念,那么可以模拟像java中将很多变量私有化封装起来,保护数据,防止数据泄漏,封装细节,这样安全性和可控性更高 function box(c ...

  8. Linux-基础学习(四)-部署图书管理系统项目

    部署图书管理项目需要以下软件 项目文件(django项目文件夹) 数据库文件(django项目对应的数据库文件) centos7(linux本体) nginx(反向代理以及静态文件收集) uWSGI( ...

  9. (三)Installation

    Elasticsearch requires at least Java 8. Specifically as of this writing, it is recommended that you ...

  10. MongoDB install

    下载地址1:https://www.mongodb.org/dl/linux下载地址2:https://www.mongodb.com/download-center/community关于Mongo ...