【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行, ...
随机推荐
- Eclipse+Maven+JDK+tomcat搭建java的开发环境
由于最近有几个同事都在学习java方面的东西,所以我写个博文做下笔记,其中遇到过很多个坑,这里就不多说了 首先,我用的是Eclipse+Maven的组合,用Ecplise是周边java开发的同事用这个 ...
- html基础 下拉菜单和文本域的基本操作
结构代码 所在城市: <select > <option selected>北京</option> <option>上海</option> ...
- php.ini配置文件中设置时区
date.timezone设置php5默认date.timezone为utc,改为date.timezone = PRC即可解决时间相差八小时的问题
- Nagios 请检查HTTP服务器关于该CGI的访问权限设置
无权查看任何主机的信息. 请检查HTTP服务器关于该CGI的访问权限设置. 搜索了一下方法 确保 htpasswd.user的所有组为nagios 解决办法: vi /usr/local/nagios ...
- Java实现163邮箱发送邮件到QQ邮箱
注:图片如果损坏,点击文章链接:https://www.toutiao.com/i6812973124141711876/ 先创建一个maven的普通项目 添加依赖,附在文档末尾 其中几个注意的地方 ...
- 基于Jenkins+Maven+Gitea+Nexus从0到1搭建CICD环境
在传统的单体软件架构中,软件开发.测试.运维都是以单个进程为单位. 当拆分成微服务之后,单个应用可以被拆分成多个微服务,比如用户系统,可以拆分成基本信息管理.积分管理.订单管理.用户信息管理.合同管理 ...
- postman设置token等关联参数
登陆时登录成功后服务器会返回一个token,这个token作为第二步骤的入参:第二个步骤请求成功后服务器会返回一个新token,然后这个token作为第三步骤的入参!如此一来的话,要用postman做 ...
- jmeter和JDK安装教程(Windows)
1.JDK的安装及环境变量配置 1.JDK的下载安装 JDK官网下载地址:https://www.oracle.com/java/technologies/downloads 然后注册账号,开始下载, ...
- Javascript中常用事件集合和事件使用方法
Javascript中常用事件集合和事件使用方法 一.事件绑定 格式: 事件源 . on事件类型=事件处理函数 事件绑定三要素 1.事件源:和谁绑定 2.事件类型:什么事件 3.事件处理函数:触发了要 ...
- golang gin框架中实现一个简单的不是特别精确的秒级限流器
起因 看了两篇关于golang中限流器的帖子: Gin 开发实践:如何实现限流中间件 常用限流策略--漏桶与令牌桶介绍 我照着用,居然没效果-- 时间有限没有深究.这实在是一个很简单的功能,我的需求是 ...