[最小生成树]vijos1579 宿命的PSS

题目梗概

输入给出最小生成树,由最小生成树求出最小完全图(任意两点之间只有一条线段相连)。

思考

首先一个图中最小生成树,按照kruskal算法。一定是使图连通的最小边。我们设这两点的边权为W,那么与两点相连的其他的边一定是W+1。

所以构建方法是 每次选出图中最短的边,其两点标记为A,B,边权为W,则与A,B连通的点边权修改为W+1。

#include <cstdio>
#include <algorithm>
typedef long long ll;

int n;
ll num[20005],fa[20005],ans; //num用来统计联通快的数量
struct node{
    ll u,v,w;
}e[20005];

bool CMP(const node &a,const node&b){
    return a.w<b.w;
}

ll find(int x){
    if(x!=fa[x]) fa[x] = find(fa[x]);
    return fa[x];
}

void merge(int x,int y,int z){
    x = find(x);
    y = find(y);
    ans += (num[x]*num[y]-1) * (z+1);   //这里是核心
    fa[y]=x;
    num[x]+=num[y];
}

void Run(){
    for(int i=1;i<=n;i++){
        fa[i]=i;
        num[i]=1;
    }
    for(int i=1;i<n;i++){
        ans+=e[i].w; //加上两点的边
        merge(e[i].u,e[i].v,e[i].w);
    }
}

void init(){
    scanf("%d",&n);
    for(int i=1;i<n;i++){
        scanf("%lld%lld%lld",&e[i].u,&e[i].v,&e[i].w);
    }
    std::sort(e+1,e+n,CMP);
}

int main(){
    init();
    Run();
    printf("%lld\n",ans);
    return 0;
}
时间: 2024-10-13 14:06:02

[最小生成树]vijos1579 宿命的PSS的相关文章

Vijos1579 宿命的PSS 最小生成树

转载一下YH大佬的blog地址http://blog.csdn.net/jokercold .. 直译小丑严寒2333 以及HK大佬 http://blog.csdn.net/yuyaohekai 宿命的PSS 题目描述 最小生成树P.S.S在宿命的指引下找到了巫师Kismi.P.S.S希望Kismi能帮自己变成一个完全图.Kismi由于某些不可告人的原因,把这件事交给了你. PS:  可以保证,这个最小生成树对于最后求出的完全图是唯一的. 输入 输入的第一行是一个整数n,表示生成树的节点数.

宿命的PSS

题目描述 最小生成树P.S.S在宿命的指引下找到了巫师Kismi.P.S.S希望Kismi能帮自己变成一个完全图.Kismi由于某些不可告人的原因,把这件事交给了你. PS:  可以保证,这个最小生成树对于最后求出的完全图是唯一的. 输入 输入的第一行是一个整数n,表示生成树的节点数. 接下来有n-1行,每行有三个正整数,依次表示每条边的端点编号和边权. (顶点的边号在1-n之间,边权< maxint) 输出 一个整数ans,表示以该树为最小生成树的最小完全图的边权之和. 样例输入 3 1 2

Linux内存VSS,RSS,PSS,USS解析

转载:http://myeyeofjava.iteye.com/blog/1837860 adb shell procrank | grep com.package > appmem说明:五个参数分别为PID Vss Rss Pss Uss 一般来说内存占用大小有如下规律:VSS >= RSS >= PSS >= USS VSS - Virtual Set Size 虚拟耗用内存(包含共享库占用的内存)RSS - Resident Set Size 实际使用物理内存(包含共享库占用

次最小生成树 模版

次小生成树(转) 转载(http://www.cnblogs.com/z360/p/6875488.html) 所谓次小生成树,顾名思义就是从生成树中取出的第二小的生成树. 我们在前面已经说过最小生成树的概念及代码实现了,所以接下来要说的次小生成树应该比较简单理解了. 求次小生成树的两种方法 1:首先求出最小生成树T,然后枚举最小生成树上的边,计算除了枚举的当前最小生成树的边以外的所有边形成的最小生成树Ti,然后求最小的Ti就是次小生成树.2:首先计算出最小生成树T,然后对最小生成树上任意不相邻

HDU1863 畅通工程---(最小生成树)

畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 27972    Accepted Submission(s): 12279 Problem Description 省政府"畅通工程"的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).经过调查评估,得到的统计表中列出

51Nod1601 完全图的最小生成树计数

传送门 我居然忘写题解啦!(记忆废) 不管怎么说,这题还算是一道好题啊--你觉得敦爷出的题会有水题么 -- 这题比较容易把人误导到Boruvka算法之类的东西上去(我们机房去刚D题的人一开始大多也被误导了),但仔细思考之后是可以发现问题的特殊性质的. 听说很多人是从Kruskal算法想到这道题的做法的?好吧我并不是,那我就写写我的思考过程好了-- 记得算导上有一道思考题,判断一个最小生成树算法的正确性.那个算法是这样的:把当前图的点集随意划分成两半,递归两半后选出连接两个点集的边中权值最小的一条

最小生成树求法 Prim + Kruskal

prim算法的思路 和dijkstra是一样的 每次选取一个最近的点 然后去向新的节点扩张 注意这里的扩张 不再是 以前求最短路时候的到新的节点的最短距离 而是因为要生成一棵树 所以是要连一根最短的连枝 所以关键部分修改一下 dist[u] = min(dist[u], e.cost) --->>e是连接 v 和 u的边 同样地 普同写法O(v^2) 用队列优化后O(E*logV) 1 #include <iostream> 2 #include <stdio.h> 3

UVALive-7303- Aquarium【最小生成树】【连通块】

UVALive - 7303- Aquarium 题目链接:7303 题目大意:给你一个r * c的格子,每个格子有一个 ' \ ' 或者 '/' 的墙,以及打掉墙的费用,问使得所有块联通的最小费用.(看图好理解) 题目思路:就是将他化成一个图,联通的块相当于一个点,两个点之间有一条边,边的权值为墙的费用. 转化为连通块的思路是:每个格子看成两部分,左侧和右侧.以一行来看,假设两个格子A,B.那么B格子的右侧的编号一定和A格子的左侧的编号相同.如图所示 给每个格子的左右侧标上号,然后加入边,边的

BZOJ_1016_[JSOI2008]_最小生成树计数_(dfs+乘法原理)

描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1016 给出一张图,其中具有相同权值的边的数目不超过10,求最小生成树的个数. 分析 生成树的计数有一个什么什么算法... 我真的企图研究了...但是智商捉急的我实在看不懂论文... 所以最后还是写了暴力... 当然暴力也要靠正确的姿势的. 首先来看一个结论: 同一张图的所有最小生成树中,边权值相同的边的数目是一定的. 也就是说,假如某一张图的某一棵最小生成树由边权值为1,1,2,2,2,3的