zoukankan      html  css  js  c++  java
  • CH 6201 走廊泼水节题解

    题目链接:CH6201

    当时在海亮考试的第一题;

    心得:其实一个算法是要真正理解这个思路和过程,而并不是单单知道它是用来写什么题的;

    思路:n个节点有n-1条边,把这n-1条边按照权值从小到大排序,有点类似Kruskal算法的过程;

    设当前扫描到边(x,y,z)时,若x,y不在同一个集合,此时应该合并Sx,Sy,此时,对于x所在集合中除x之外的点u,y所在集合中除y之外的点v,完全图中u与v之间肯定要连一条边,所以共同构成一个环,因为要保证边(x,y)一定在最小生成树中,就必须让(x,y)是连接两个集合的边权最小的边。设x,y之间权值为z,所以(u,v)的边权最小为z+1。而Sx与Sy之间最后一共会增加(size[x]*size[y]-1)条边,所以把(z+1)*(size[x]*size[y]-1)累加到答案中;

    算法时间复杂度O(NlogN);

    #include<bits/stdc++.h>
    using namespace std;
    #define N 500001
    template<typename T>inline void read(T &x)
    {
        x=0;T f=1,ch=getchar();
        while(!isdigit(ch))  {if(ch=='-')  f=-1;  ch=getchar();}
        while(isdigit(ch))  {x=x*10+ch-'0';  ch=getchar();}
        x*=f;
    }
    struct gg
    {
        int x,y,v;
    }a[N<<1];
    int father[N],size[N],t,n;
    bool cmp(gg x,gg y){return x.v<y.v;}
    inline int find(int x){return father[x]==x?x:father[x]=find(father[x]);}
    int main() {
        read(t);
        while(t--) {
            read(n);
            for(int i=0;i<=n;i++)    father[i]=i,size[i]=1;
            for(int i=1;i<n;i++)
                read(a[i].x),read(a[i].y),read(a[i].v);
            sort(a,a+n,cmp);
            long long ans=0;
            for(int i=1;i<n;i++) {
                int p=find(a[i].x),q=find(a[i].y);
                if(p==q) continue;
                ans+=(long long)(a[i].v+1)*(size[p]*size[q]-1);
                father[p]=q;
                size[q]+=size[p];
            }
            printf("%lld
    ",ans);
        }
        return 0;
    }
  • 相关阅读:
    Windows10如何添加开机启动项
    selenium自动化文件上传、文件下载
    windows下创建虚拟环境
    selenium 处理js日历控件
    selenium鼠标和键盘事件
    selenium绕过某宝验证,手动验证滑块
    python国内镜像源
    Linux下jenking安装
    发送邮件
    基于python的selenium两种文件上传操作
  • 原文地址:https://www.cnblogs.com/Tyouchie/p/10667370.html
Copyright © 2011-2022 走看看