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;
    }
  • 相关阅读:
    作业2
    实验12——指针的基础应用2
    实验11——指针的基础应用
    实验十——一维数组的定义及引用
    实验九——基本数据类型存储及应用总结
    实验八--函数定义及调用总结
    实验七——函数定义及调用总结
    实验六——循环结构程序练习总结
    实验五——循环结构学习总结
    实验三——for 语句及分支结构else-if
  • 原文地址:https://www.cnblogs.com/Tyouchie/p/10667370.html
Copyright © 2011-2022 走看看