【图论】CF1508C Complete the MST
有一张 \(n\) 个点的完全图,其中 \(m\) 条边已经标有边权。你需要给剩下的边都标上权值,使得所有边权的异或和为 \(0\),并且整张图的最小生成树边权和最小。
\(n,m\le 10^6, 1\le w_i<2^{30}\)。
注意到这道题是要让最小值最小,所以全程的决策都是由我们来决定的。
设已染色的边权和为 \(S\),那可以发现不会染两条以上的边,理由是 \(x+y\ge x\oplus y\)。只用考虑染一条 \(S\)。
记 \(ans'\) 为视每个补图连通块为一个点,最小生成树边权和。考虑什么情况下这条 \(S\) 边压根不会被算入答案。当所有未确定的边不构成森林的时候答案一定是 \(ans'\),因为随便弄出一棵最小生成树都一定有一条边不在里面,染成 \(S\) 即可。
否则最小生成树是森林,此时点数已经是 \(O(\sqrt m)\) 级别,可以直接枚举哪条边是 \(S\),时间复杂度 \(O(m\sqrt m)\)。如果这么做这道题就没意思了。
继续推性质。考虑我们有没有比 \(ans'+S\) 更优的方案。首先拿出我们当前的最小生成树,称其为 \(T\),则如果我们能省下一条补图的边不被选(用来放 \(S\)),那么在这棵补图树中割掉这条边后,一定会有恰好一边没在当前连通块中,所以需要新增恰好一条原图的边;反之,如果新增一条原图的边,且这条边的两端在 \(T\) 中不连通,则一定存在一条补图的树边可以被割掉。这两者形成一种双射。
所以从小到大枚举每条边看两个端点连不连通即可。
至于如何求出补图连通块,我们维护一个链表为当前还没有选的点,每次拿出链头,表示找到一个新的连通块,然后进行 bfs,每次对于一个点 \(u\),暴力遍历在链表中的所有点,如果不和 \(u\) 有连边则从链表中删除加入队列。由于每个点至多被删一次,并且每条边至多被访问一次而贡献时间复杂度,所以这部分是均摊线性的。
总时间复杂度 \(O(m\log m)\),瓶颈在于排序和用 set 判断边是否存在。
点击查看代码
#include <bits/stdc++.h>
#define For(i,a,b) for(int i=a;i<=b;i++)
#define Rev(i,a,b) for(int i=a;i>=b;i--)
#define Fin(file) freopen(file,"r",stdin);
#define Fout(file) freopen(file,"w",stdout);
using namespace std;
const int N=2e5+5; using ll = long long;
struct Edge{int x,y,z;bool operator<(const Edge& ed)const{return z<ed.z;};}ed[N]; int ecnt;
int n,m,cp[N],ccnt,siz[N],ec[N],fa[N],flg[N]; set<int> S[N];
int getfa(int x) { return x==fa[x]?x:fa[x]=getfa(fa[x]); }
int main(){
cin>>n>>m; int s=0; For(i,1,m) { int x,y,z; cin>>x>>y>>z; s^=z; ed[++ecnt]={x,y,z}; S[x].insert(y),S[y].insert(x); }
sort(ed+1,ed+1+ecnt); list<int> lis; For(i,1,n) lis.push_back(i);
while(lis.size()){
ccnt++; queue<int> q; q.push(lis.front()); lis.pop_front();
while(q.size()){
int u=q.front(); q.pop(); cp[u]=ccnt;
for(auto it=lis.begin();it!=lis.end();){
if(!S[u].count(*it)) q.push(*it),lis.erase(it++); else it++;
}
}
}
For(u,1,n) { siz[cp[u]]++; for(int v:S[u]) if(cp[v]==cp[u]) ec[cp[u]]++; }
ll ans=0; int add=s; iota(fa+1,fa+1+ccnt,1);
For(i,1,ecnt){
int x=getfa(cp[ed[i].x]),y=getfa(cp[ed[i].y]); if(x!=y) flg[i]=1,ans+=ed[i].z,fa[y]=x;
}
For(i,1,ccnt) if(ec[i]/2+siz[i]-1!=1ll*siz[i]*(siz[i]-1)/2) cout<<ans<<'\n',exit(0);
iota(fa+1,fa+1+n,1); For(i,1,ecnt) if(flg[i]) fa[getfa(ed[i].y)]=getfa(ed[i].x);
For(i,1,ecnt) if(ed[i].z<add) {
int x=ed[i].x,y=ed[i].y; if(getfa(x)!=getfa(y)) { add=ed[i].z; break; }
}
cout<<ans+add<<'\n';
return 0;
}
【图论】CF1508C Complete the MST的更多相关文章
- 算法对比:Prim算法与Dijskra算法
在图论中,求MST的Prim算法和求最短路的Dijskra算法非常像.可是我一直都对这两个算法处于要懂不懂的状态,现在,就来总结一下这两个算法. 最小生成树(MST)—Prim算法: 算法步骤: •将 ...
- Codeforces Round #599 (Div. 1) B. 0-1 MST 图论
D. 0-1 MST Ujan has a lot of useless stuff in his drawers, a considerable part of which are his math ...
- 【图论 思维】cf715B. Complete The Graph加强
zzq讲的杂题 题目大意 有一张$n$个点$m$条边的简单正权无向图,$S$到$T$的最短路为$L$,现在有一些边的边权未知,请输出任意一种满足题意的方案. $n,m\le 500000$ ...
- NOI.AC 31 MST——整数划分相关的图论(生成树、哈希)
题目:http://noi.ac/problem/31 模拟 kruscal 的建最小生成树的过程,我们应该把树边一条一条加进去:在加下一条之前先把权值在这一条到下一条的之间的那些边都连上.连的时候要 ...
- D. Edges in MST 图论
http://codeforces.com/contest/160/problem/D base on 克鲁斯卡尔, 首先每次都是对权值相同的边进行统一处理,假如加入了当前这条边出现了回路,那就能确定 ...
- 图论 --- BFS + MST
Borg Maze Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 7844 Accepted: 2623 Descrip ...
- poj 1679 The Unique MST 判断最小生成树是否唯一(图论)
借用的是Kruskal的并查集,算法中的一点添加和改动. 通过判定其中有多少条可选的边,然后跟最小生成树所需边做比较,可选的边多于所选边,那么肯定方案不唯一. 如果不知道这个最小生成树的算法,还是先去 ...
- [转] POJ图论入门
最短路问题此类问题类型不多,变形较少 POJ 2449 Remmarguts' Date(中等)http://acm.pku.edu.cn/JudgeOnline/problem?id=2449题意: ...
- HDU5627--Clarke and MST (bfs+位运算)
http://www.cnblogs.com/wenruo/p/5188495.html Clarke and MST Time Limit: 2000/1000 MS (Java/Others) M ...
- 一个完整的成年果蝇大脑的电子显微镜图谱 | A Complete Electron Microscopy Volume of the Brain of Adult Drosophila melanogaster
paper:A Complete Electron Microscopy Volume of the Brain of Adult Drosophila melanogaster 果蝇是一个非常完美的 ...
随机推荐
- 零代码修改,教你Spring Cloud应用轻松接入CSE
摘要:本文介绍了Sermant Agent的接入原理和如何使用Sermant Agent无修改接入CSE. 本文分享自华为云社区<Spring Cloud应用零代码修改接入华为云微服务引擎CSE ...
- 聊聊数仓中TPCD-DS&TPC-H与查询性能的那些事儿
摘要:详细讲述使用GaussDB(DWS)时,如何使用TPC-DS/TPC-H等标准数据模型,获取DWS的查询性能数据. 本文分享自华为云社区<GaussDB(DWS) <DWS之TPCD ...
- MongoDB 占用CPU资源过高
情况如下 db.currentOp() 发现有全表扫描 将 Collscan 对应的 Collection 建索引 db.Table1.createIndex({"DataTime" ...
- 将MyBatis Mapper xml 放到 jar 包外面
在不改程序的情况下,修改 sql 时,需要将 Mapper 中的 XML 文件 放到外面 mybatis: mapper-locations: classpath:mapper/*.xml #J ...
- compilation.templatesPlugin is not a function
ERROR Failed to compile with 1 error TypeError: compilation.templatesPlugin is not a function - SetV ...
- 【Django drf】认证类 权限类 频率类 过滤类 排序类 分页类
目录 认证类 前期准备 自定义认证类 配置认证类 全局配置 局部禁用 认证组件使用步骤 基于类中方法的认证(了解) 权限类 前期准备 重写has_permission() 添加权限不足信息 权限组件使 ...
- EasyUI DataGrid 没有数据
//判断后台返回数据是否没数据,没数据DataGrid添加一行 $(this).datagrid('appendRow', { itemid: '<div style="text-al ...
- anaconda学习(未完成)
1.Anaconda安装教程(以32.7.4为例)官网地址:https://www.anaconda.com/download(如无法下载可跳转清华源下载)下载完成后点击打开即可安装点击Next选择I ...
- # github.com/coreos/etcd/clientv3/balancer/resolver/endpoint
linux使用go连接etcd集群时报错: # github.com/coreos/etcd/clientv3/balancer/resolver/endpoint /root/go/pkg/mod/ ...
- 悟空活动中台 - 基于 WebP 的图片高性能加载方案
本文首发于 vivo互联网技术 微信公众号 链接: https://mp.weixin.qq.com/s/rSpWorfNTajtqq_pd7H-nw作者:悟空中台研发团队 一.背景 移动端网页的加载 ...