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;
    }
  • 相关阅读:
    Fetch的使用
    if判断中的true和false
    分布式、微服务和集群的初步了解
    关于视频的知识点
    ajax请求
    jq的遍历关系元素方法集合
    docker安装Mysql
    设计模式系列之七大原则之——开闭原则
    设计模式系列之七大原则之——里式替换原则
    设计模式系列之七大原则之——依赖倒转原则
  • 原文地址:https://www.cnblogs.com/Tyouchie/p/10667370.html
Copyright © 2011-2022 走看看