CF1408E Avoid Rainbow Cycles 题解
解题思路
第一眼看过去感觉不是很可做……
但是我们可以发现,如果有两个点在不同的集合中出现过,那么一定会存在彩虹环,那么两个点最多出现一次。同时我们考虑将题意转化一下,变成求最大能选取的点,使得不出现彩虹环。根据刚刚的性质,我们可以考虑每个点向它所在的集合连一条边权为
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);
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· Manus爆火,是硬核还是营销?
· 一文读懂知识蒸馏
· 终于写完轮子一部分:tcp代理 了,记录一下