【BZOJ2654】tree(生成树 二分)
大意
给你一个无向带权连通图,每条边是黑色或白色,求一棵最小权的恰好有\(Need\)条白色边的生成树。
题目保证有解,输出最小权值。
其中每条边权在\([1,100]\)范围内。
思路
首先有一个比较明显的想法:
用Kruskal跑出一个最小生成树,然后再不断往其中加边调整白色边的数量,用LCT维护圈内最大异色边。
好吧,这种极其复杂的算法可以被以下例子卡掉。
假如随便跑的一个最小生成树是下图:
(边左边为颜色,右边为边权)

再考虑加入以下边:

假如需要一条白边,并且按从小到大的顺序选白边,那么就会先选 6-7 这条边,删 5-8 这条边,这样做会产生 1 的贡献。但如果我们选 2-3 这条边,删 1-4 这条边,那么对答案的贡献就是 0,明显更优。
考虑一个正常的算法:
我们考虑给每条白边附加一个权值\(W\),使得白边边权由\(Val\)变为\(Val+W\),然后再跑一遍最小生成树。
可以发现,当\(W\)越大时,白边数量越少,即呈单调性。
于是考虑二分\(W\)的值,每次二分根据当前最小生成树能得到的最小或最大白边数与\(Need\)的大小关系Check就行。
正确性小记:
- 对于相邻的两个\(W\)值,倘若存在它们的可选白边数量区间正好不等(即无交集),而\(Need\)又正好在它们中间的情况,那么此时就会出问题(无法二分到正确解)。
- 但其实这种情况并不会出现,设两个区间的空隙为\(D(D\ge 2)\),那么就会有至少\(D\)条白边在刚才的+1中,变得比原位上的黑边大1,设原位上的黑边数量为\(X(X\ge 1)\)。

- 设左边那个区间的右端点为\(R\)(如图),那么在算\(L\)的情况时,那\(X\)条黑边的优先级就会比\(D\)条白边的优先级高;同时,在算\(R\)的情况下也是如此。故在算\(L\)与\(R\)时,Kruskal算法中边的顺序是一样的,即相邻两个区间是会重着端点的,即\(D=0\),即不存在\(Need\)不被任意一个区间包含的情况。

细节:
- 二分取值与最小生成树所取最小最大白边数的关系:
由于对于同一个\(W\),可能会有多种选白边数量的方法而生成相同权值的最小生成树的情况,即白边可选数量实际上是一段区间,所以我们需要考虑是取该区间的左端点还是右端点。 - 倘若是从下界逼近,即\(Ans\)选取\(L\)时,那么应取最小白边数量,这样\(L\)才会在 \(Mid\)正好为答案且\(Mid\)的白边数量区间跨越了\(Need\) 的情况下合法(\(R\)不会占\(L\)的位置)。
- 同理,\(Ans\)选取\(R\)时,应取最大白边数量。
TLE小记:
- 时间复杂度为\(O(N*log(N)*log(W))\),其中\(log(W)\)决定着代码的命运。
代码
#include<cstdio>
#include<algorithm>
using namespace std;
const int MAXN=100005;
int N,M,Ned,Fa[MAXN];long long Ans;
struct Edge{int x,y,col,vis,id,z;}s[MAXN],tmp[MAXN];
bool cmp(Edge A,Edge B){return A.z<B.z||(A.z==B.z&&A.col<B.col);}
int Find(int x){return Fa[x]==x?x:Fa[x]=Find(Fa[x]);}
void Turn(double p){
for(int i=1;i<=M;i++){
s[i]=tmp[i];
if(tmp[i].col==0)s[i].z+=p;
}sort(s+1,s+M+1,cmp);
}
int Get(){
int ret=0;
for(int i=1;i<=N;i++)Fa[i]=i;
for(int i=1;i<=M;i++){
int x=Find(s[i].x),y=Find(s[i].y);
if(x==y)continue;Fa[x]=y;
if(s[i].col==0)ret++;
s[i].vis=1;
}
return ret;
}
bool Check(int p){
Turn(p);return Get()<=Ned;
}
int main(){
scanf("%d%d%d",&N,&M,&Ned);
for(int i=1,col;i<=M;i++)
scanf("%d%d%d%d",&tmp[i].x,&tmp[i].y,&tmp[i].z,&tmp[i].col),tmp[i].x++,tmp[i].y++,tmp[i].id=i;
double L=-101,R=101;
while(L+1<R){
double mid=(L+R)/2;
if(Check(mid))R=mid;
else L=mid;
}
Turn(L);Get();Ans=0;
for(int i=1;i<=M;i++)
if(s[i].vis)Ans+=tmp[s[i].id].z;
printf("%lld\n",Ans);
}
【BZOJ2654】tree(生成树 二分)的更多相关文章
- BZOJ2654:tree(最小生成树,二分)
Description 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树. 题目保证有解. Input 第一行V,E,need分别表示点数,边数和需要的白色 ...
- BZOJ2654 tree 【二分 + 最小生成树】
题目 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树. 题目保证有解. 输入格式 第一行V,E,need分别表示点数,边数和需要的白色边数. 接下来E行, ...
- [BZOJ2654]tree(二分+MST)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2654 分析:此题很奇葩,我们可以给所有白边加上一个权值mid,那么在求得的MST中白边 ...
- [BZOJ2654] tree (kruskal & 二分答案)
Description 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树. 题目保证有解. Input 第一行V,E,need分别表示点数,边数和需要的白色 ...
- 2021.07.19 BZOJ2654 tree(生成树)
2021.07.19 BZOJ2654 tree(生成树) tree - 黑暗爆炸 2654 - Virtual Judge (vjudge.net) 重点: 1.生成树的本质 2.二分 题意: 有一 ...
- [BZOJ2654]tree(二分+Kruskal)
2654: tree Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 2733 Solved: 1124[Submit][Status][Discus ...
- BZOJ2654: tree 二分答案+最小生成树
Description 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树. 题目保证有解. Input 第一行V,E,need分别表示点数,边数和需要的白色 ...
- [BZOJ2654]:tree(Kruskal+WQS二分)
题目传送门 题目描述 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树.题目保证有解. 输入格式 开始标号),边权,颜色(0白色1黑色). 输出格式 一行表 ...
- [bzoj2654] tree 最小生成树kruskal+二分
题目描述 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树.题目保证有解. 输入格式 第一行V,E,need分别表示点数,边数和需要的白色边数.接下来E行, ...
随机推荐
- 实践剖析.NET Core如何支持Cookie和JWT混合认证、授权
前言 为防止JWT Token被窃取,我们将Token置于Cookie中,但若与第三方对接,调用我方接口进行认证.授权此时仍需将Token置于请求头,通过实践并联系理论,我们继续开始整活!首先我们实现 ...
- Kubernetes-Kuboard
前言 本篇是Kubernetes第十五篇,大家一定要把环境搭建起来,看是解决不了问题的,必须实战. Kubernetes系列文章: Kubernetes介绍 Kubernetes环境搭建 Kubern ...
- [服务器部署] Flask + virtualenv + uWSGI + Nginx 遇到的问题
1.配置好了Flask + virtualenv +uWSGI,启动uWSGI并调试,网页显示 Internal Server Error 参考:https://www.cnblogs.com/cle ...
- 经典变长指令-ModRM
一.如何计算ModRM 0X88 MOV Eb,Gb G:通用寄存器 0X89 MOV Ev,Gv E:寄存器/内存 0X8A MOV Gb,Eb b:字节 0X8B MOV Gv,Ev v:Word ...
- vue3+vant h5: Rem 移动端布局适配之postcss-pxtorem和lib-flexible
如果不引入插件的话:ui稿的px转化成rem需自己计算 根据设计稿我们需要自己计算元素的rem(假如我们将html根元素font-size设置为41.4px): 那么1rem=41.4px; ui稿上 ...
- 【reverse】逆向6 JCC
[reverse]逆向6 JCC 前言 我们之前学习的时候讲了,eip寄存器存储的是当前(即将执行的语句的) 指向地址 而我们之前提到的下断点(F2),就和我们编程中的下断点一样,执行到某句汇编指令然 ...
- win+ r 命令
Win 键+R calc:计算器 notepad:记事本 mspaint:画图 cmd:控制台 control:控制面板 desk.cpl:打开控制面板中的桌面设置 main.cpl:鼠标设置 ine ...
- RichTextBox单独设置文字颜色
richTextBox1.Select(index, "str1".Length); richTextBox1.SelectionColor = Color.Red;
- 【练习】rust中的复制语义和移动语义
1.基本类型都是复制语义的 fn main(){ let a = 123; { #[allow(unused_variables)] let b = a; //如果是移动语义,那么后续的a将不再有效 ...
- Android 12(S) 图形显示系统 - 应用建立和SurfaceFlinger的沟通桥梁(三)
1 前言 上一篇文章中我们已经创建了一个Native示例应用,从使用者的角度了解了图形显示系统API的基本使用,从这篇文章开始我们将基于这个示例应用深入图形显示系统API的内部实现逻辑,分析运作流程. ...