Title

CF1408E Avoid Rainbow Cycles 题解

解题思路

第一眼看过去感觉不是很可做……

但是我们可以发现,如果有两个点在不同的集合中出现过,那么一定会存在彩虹环,那么两个点最多出现一次。同时我们考虑将题意转化一下,变成求最大能选取的点,使得不出现彩虹环。根据刚刚的性质,我们可以考虑每个点向它所在的集合连一条边权为 ai+bj 的边,那么我们最终要选取一些边使得这些边的边权和最大且不存在两条边在多个集合中出现,即联通块中无环。那么这个时候就很自然的想到最大生成树了,我们每次选择最大的能选的边并 sumsum+wi,那么我们最后的答案即为 (i=1j=1msjwi)sum

AC 代码

#include<stdio.h>
#include<stdlib.h>
#include<algorithm>
#define int long long
#define N 200005
int n,m,k;
int a[N],b[N];
struct Edge{
    int u,v,w;
}edge[N<<1];
inline bool cmp(Edge x,Edge y){
    return x.w>y.w;       
}int fa[N<<1],ce;
inline int find(int x){
    return x==fa[x]?x:
    fa[x]=find(fa[x]);
}
signed main(){int ans=0;
    scanf("%lld%lld",&m,&n);
    for(register int i=1;i<=m;++i)
        scanf("%lld",&a[i]);
    for(register int i=1;i<=n;++i)
        scanf("%lld",&b[i]);
    for(register int i=1;i<=m;++i){
        scanf("%lld",&k);int x;
        for(register int j=1;j<=k;++j){
            scanf("%lld",&x);
            edge[++ce]={x,i+n,a[i]+b[x]};
            ans+=edge[ce].w;
        }
    }for(register int i=0;i<=n+m;++i)
        fa[i]=i;
    std::sort(edge+1,edge+ce+1,cmp);
    for(register int i=1;i<=ce;++i){
        int fx=find(edge[i].u);
        int fy=find(edge[i].v);
        if(fx==fy) continue;
        ans-=edge[i].w;fa[fx]=fy;
    }printf("%lld",ans);
}
posted @   UncleSam_Died  阅读(3)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· Manus爆火,是硬核还是营销?
· 一文读懂知识蒸馏
· 终于写完轮子一部分:tcp代理 了,记录一下
点击右上角即可分享
微信分享提示