Problem Link

有一张 \(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的更多相关文章

  1. 算法对比:Prim算法与Dijskra算法

    在图论中,求MST的Prim算法和求最短路的Dijskra算法非常像.可是我一直都对这两个算法处于要懂不懂的状态,现在,就来总结一下这两个算法. 最小生成树(MST)—Prim算法: 算法步骤: •将 ...

  2. 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 ...

  3. 【图论 思维】cf715B. Complete The Graph加强

    zzq讲的杂题 题目大意 有一张$n​$个点$m​$条边的简单正权无向图,$S​$到$T​$的最短路为$L​$,现在有一些边的边权未知,请输出任意一种满足题意的方案. $n,m\le 500000​$ ...

  4. NOI.AC 31 MST——整数划分相关的图论(生成树、哈希)

    题目:http://noi.ac/problem/31 模拟 kruscal 的建最小生成树的过程,我们应该把树边一条一条加进去:在加下一条之前先把权值在这一条到下一条的之间的那些边都连上.连的时候要 ...

  5. D. Edges in MST 图论

    http://codeforces.com/contest/160/problem/D base on 克鲁斯卡尔, 首先每次都是对权值相同的边进行统一处理,假如加入了当前这条边出现了回路,那就能确定 ...

  6. 图论 --- BFS + MST

    Borg Maze Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7844   Accepted: 2623 Descrip ...

  7. poj 1679 The Unique MST 判断最小生成树是否唯一(图论)

    借用的是Kruskal的并查集,算法中的一点添加和改动. 通过判定其中有多少条可选的边,然后跟最小生成树所需边做比较,可选的边多于所选边,那么肯定方案不唯一. 如果不知道这个最小生成树的算法,还是先去 ...

  8. [转] POJ图论入门

    最短路问题此类问题类型不多,变形较少 POJ 2449 Remmarguts' Date(中等)http://acm.pku.edu.cn/JudgeOnline/problem?id=2449题意: ...

  9. HDU5627--Clarke and MST (bfs+位运算)

    http://www.cnblogs.com/wenruo/p/5188495.html Clarke and MST Time Limit: 2000/1000 MS (Java/Others) M ...

  10. 一个完整的成年果蝇大脑的电子显微镜图谱 | 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 果蝇是一个非常完美的 ...

随机推荐

  1. 云图说丨应用宕机怎么办?MAS帮您实现业务无缝切换

    摘要: 多云高可用服务(Multi-cloud high Availability Service,简称MAS)源自华为消费者多云应用高可用方案,提供从流量入口.数据到应用层的端到端的业务故障切换及容 ...

  2. 最新的iOS应用上架App Store详细流程解析

    最新的iOS应用上架App Store详细流程解析 2023已经过了2/3的时间,由于现在苹果签名市场的价格不断的上升,现在很多的开发商一直在想着如何进行上架一些自己的产品,下面小编来给大家梳理一下上 ...

  3. 【主流技术】MongoTemplate 与 Spring Boot 项目集成分享(附CURD技巧)

    目录 前言 一.表结构特点 1.1Json格式 1.2实体映射 二.条件构造 2.1Criteria与Query的区别 2.2简单条件 2.3复杂条件 三.如何选用接口 3.1MongoReposit ...

  4. Zookeeper面试题总结

    1.请简述Zookeeper的选举机制 假设有五台服务器组成的zookeeper集群,它们的id从1-5,同时它们都是最新启动的,也就是没有历史数据,在存放数据量这一点上,都是一样的. 假设这些服务器 ...

  5. Python使用pandas库读取txt文件中的Json数据,并导出到csv文件

    使用的是 Python 3 代码: #!/usr/bin/python # -*- coding: gbk -*- # JSON数据导出到csv文件 import json import pandas ...

  6. Windows10/11 wsl2 安装 ArchLinux 子系统

    这篇文章针对的是在win11系统的wsl2下安装ArchLinux系统,网上很多中文教程都是使用LxRunOffline去做的,但是实际上该方法已经过时了,目前有更加先进的ArchWSL方式. 如果用 ...

  7. L2-020 功夫传人 (25分)

    分析: ⽤⼆维数 组v存储师⻔谱系关系,v[i]表示编号为i的师傅所拥有的徒弟,如果徒弟个数等于0, 也就是说这是个得道者,那么v[i][0]保存放⼤的倍数,⽽且⽤visit[i] = true标记当 ...

  8. poi4版本处理word里表格中的文字换行问题和设置字体样式

    开发中遇到生成word文档的需求,其中里面存在表格,发现表格中一旦存在换行生成的模板就出现各种问题,反正就是出不来想要的结果.网上找了一些方法基本都不好用,最后找到一个靠谱点的方法 XWPFParag ...

  9. S3C2440移植linux3.4.2内核之内核裁剪

    上一节S3C2440移植linux3.4.2内核之支持YAFFS文件系统我们修改了内核支持了yaffs2文件系统,这节我们裁剪内核. 目录 为什么要裁剪内核? 首先裁剪内核里无关的CPU/单板文件 裁 ...

  10. localstorage、sessionstorage 存储布尔值要注意

    今天实现一个功能,点击首页,头部按钮,切换大屏功能.不同的屏,页头宽度不一样,小屏1280px,  大屏百分百屏幕,左右留点120px的padding值. 思路:点击按钮,切换不同的子路由,顶部通过一 ...