[国家集训队2012]tree(陈立杰)
[国家集训队2012]tree(陈立杰)
题目
给你一个无向带权连通图,每条边是黑色或白色。让你求一棵最小权的恰好有need条白色边的生成树。
题目保证有解。INPUT
第一行V,E,need分别表示点数,边数和需要的白色边数。
接下来E行
每行s,t,c,col表示这边的端点(点从0开始标号),边权,颜色(0白色1黑色)。OUTPUT
一行表示所求生成树的边权和。
SAMPLE
INPUT
2 2 1
0 1 1 1
0 1 2 0
OUTPUT
2
数据规模
0:V<=10
1,2,3:V<=15
0,..,19:V<=50000,E<=100000
所有数据边权为[1,100]中的正整数。
解题报告
国家集训队的题,果然是好题= =
首先,我们观察题面,显然与最小生成树有什么关系,但是,直接跑最小生成树显然也是不合理的,那么问题就在于,如何在跑最小生成树的同时,还能保证白边的个数?
正解是个很神奇的东西——二分
首先我们想怎么二分,我们注意到,边权的范围很小,是$[1,100]$,那么,我们是否可以通过控制白边的边权,达到在最小生成树中控制白边的数量呢?
显然可以。
我们以$Kruskal$算法为例,我们进行$Kruskal$时,是以每条边的边权为依据,进行从小到大排序,然后从小到大取出各个边,不断加入连通分量中,最后形成最小生成树。那么,当我们改变某一些边的权值时,我们按权值排序得出的边的序列也一定就会不一样,那么,我们就可以通过控制权值来控制加入生成树的白边数了。
具体做法:
在$[-100,100]$中二分得到$mid$,让所有白边的权值加上该$mid$值,跑$Kruskal$,直到得到最终结果
但是,只是这样就可以了吗?
显然不是。
我们考虑,假如我们点很少,只有$10+$个点,但是我们的边很多,达到了$100000$,而且权值范围还被限制在了$[1,100]$,那么显然,会有许多等价的黑边与白边,即使加上了某一个权值,也可能会有很多白边与很多黑边相等价,当我们按照权值排序的同时,我们把黑边与白边混在了一起,然后我们就开始了$Kruskal$,那样的话,我们本来可以得到刚好$need$条白边,我们却把一些与黑边等价的白边扔进了生成树中,这样的话,我们本来可以得到最优解,却认为它是不合法的。
这时我们就需要处理一下这些等价的边。
具体做法:
在二分后判断时,我们不单单判断此时白边的数目是否等于$need$,而是将一切白边数大于等于$need$的情况全部考虑上,然后,我们加上了(或者是减去,因为我们有负数)许多为了控制生成树的权值,所以我们需要减去(或加上)这些权值,我们并不能将所有白边修改的权值修改回去,而是将$need\times mid$权值处理掉,因为假如我们处理所有的白边,我们可能将那些等价于黑边的白边也处理掉了,也就是相当于处理了黑边,显然是不合法的,所以我们只处理$need$条,也就是真正被当成白边的白边数量
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
inline int read(){
int sum(),f();
char ch(getchar());
for(;ch<''||ch>'';ch=getchar())
if(ch=='-')
f=-;
for(;ch>=''&&ch<='';sum=sum*+(ch^),ch=getchar());
return sum*f;
}
struct edge{
int s,e,w,col,tmp;
friend bool operator<(const edge &a,const edge &b){
return a.tmp==b.tmp?a.col<b.col:a.tmp<b.tmp;
}
}a[];
int n,m,need;
int fa[];
inline int find(int x){
if(fa[x]==x)
return x;
fa[x]=find(fa[x]);
return fa[x];
}
int tp;
inline bool krus(){
tp=;
int tot(),whit();
sort(a+,a+m+);
for(int i=;i<=m;++i){
int s(a[i].s),e(a[i].e);
int fs(find(s)),fe(find(e));
if(fs!=fe){
fa[fe]=fs;
if(a[i].col==)
++whit;
++tot;
tp+=a[i].tmp;
if(tot==n-)
break;
}
}
return whit>=need;
}
inline int gg(){
freopen("nt2012_tree.in","r",stdin);
freopen("nt2012_tree.out","w",stdout);
n=read(),m=read(),need=read();
for(int i=;i<=m;++i)
a[i].s=read()+,a[i].e=read()+,a[i].w=read(),a[i].col=read();
int l(-),r(),ans;
while(l<=r){
int mid((l+r)>>);
for(int i=;i<=n;++i)
fa[i]=i;
for(int i=;i<=m;++i){
if(a[i].col==)
a[i].tmp=a[i].w+mid;
else
a[i].tmp=a[i].w;
}
if(krus())
l=mid+,ans=tp-need*mid;
else
r=mid-;
}
printf("%d",ans);
return ;
}
int K(gg());
int main(){;}
[国家集训队2012]tree(陈立杰)的更多相关文章
- [国家集训队2012]tree(陈立杰) 题解(二分+最小生成树)
tree 时间限制: 3 Sec 内存限制: 512 MB 题目描述 给你一个无向带权连通图,每条边是黑色或白色.让你求一棵最小权的恰好有need条白色边的生成树. 题目保证有解. 输入 第一行V, ...
- [国家集训队2012]middle(陈立杰)
我是萌萌的传送门 我是另一个萌萌的传送门 脑残错误毁一下午…… 其实题解早就烂大街了,然而很久之前我只知道是二分答案+主席树却想不出来这俩玩意儿怎么一块儿用的……今天又翻了几篇题解才恍然大悟,是把权值 ...
- 数据结构(动态树):[国家集训队2012]tree(伍一鸣)
[问题描述] 一棵n个点的树,每个点的初始权值为1.对于这棵树有q个操作,每个操作为以下四种操作之一: + u v c:将u到v的路径上的点的权值都加上自然数c: - u1 v1 u2 v2:将树中原 ...
- [COGS 1799][国家集训队2012]tree(伍一鸣)
Description 一棵n个点的树,每个点的初始权值为1.对于这棵树有q个操作,每个操作为以下四种操作之一: + u v c:将u到v的路径上的点的权值都加上自然数c: - u1 v1 u2 v2 ...
- cogs1799 [国家集训队2012]tree(伍一鸣)
LCT裸题 注意打标记之间的影响就是了 这个膜数不会爆unsigned int #include<cstdio> #include<cstdlib> #include<a ...
- [国家集训队2012]middle
http://cogs.pro:8080/cogs/problem/problem.php?pid=1763 二分答案x 把区间内>=x的数设为1,<x的数设为-1 左端点在[a,b]之间 ...
- luogu P2619 [国家集训队2]Tree I
题目链接 luogu P2619 [国家集训队2]Tree I 题解 普通思路就不说了二分增量,生成树check 说一下坑点 二分时,若黑白边权有相同,因为权值相同优先选白边,若在最有增量时出现黑白等 ...
- [国家集训队2012]JZPFAR
[国家集训队2012]JZPFAR 题目 平面上有n个点.现在有m次询问,每次给定一个点(px, py)和一个整数k,输出n个点中离(px, py)的距离第k大的点的标号.如果有两个(或多个)点距离( ...
- P2619 [国家集训队2]Tree I(最小生成树+二分)
P2619 [国家集训队2]Tree I 每次二分一个$x$,每条白边加上$x$,跑最小生成树 统计一下满足条件的最小值就好了. to me:注意二分不要写挂 #include<iostream ...
随机推荐
- JSP-Runoob:JSP 点击量统计
ylbtech-JSP-Runoob:JSP 点击量统计 1.返回顶部 1. JSP 点击量统计 有时候我们需要知道某个页面被访问的次数,这时我们就需要在页面上添加页面统计器,页面访问的统计一般在用户 ...
- Spark SQL中 RDD 转换到 DataFrame (方法二)
强调它与方法一的区别:当DataFrame的数据结构不能够被提前定义.例如:(1)记录结构已经被编码成字符串 (2) 结构在文本文件中,可能需要为不同场景分别设计属性等以上情况出现适用于以下方法.1. ...
- Kubernetes 集群中使用 Helm 搭建 Spinnaker
在我们部署Spinnaker之前,我们需要一个YAML格式的配置文件,它会包含了一些配置信息.可以从Spinnaker Helm Chart repository[2]获得这个文件. $curl -L ...
- MAC地址 初识
MAC地址 即物理地址/硬件地址 地址长度为48位,6字节. 格式为:00-23-5A-15-99-42 一个网卡对应一个MAC地址(比如笔记本,有线网卡有一个MAC地址,无线网卡也有一个MAC地址) ...
- win10系统下,开启数据库远程连接方式
右键左下角的windows标志,选择控制面板 2.查看方式修改为大图标 3.选择高级设置 4.新建入站规则 5.选择端口然后下一步 6.选择tcp协议,端口输入80,3306 7.选择允许连接 8.规 ...
- MVVMLight消息通知实现机制详解(二)
接上文 MVVMLight消息通知实现机制详解(一) 该工具的内部主要逻辑是以字典模式进行储存持有订阅对象设置的传入参数Type类型.Key值.Action.Target(订阅对象本身) 在发生订阅事 ...
- MSXML2.XMLHTTP.4.0对象
一.使用步骤:1.创建XMLHTTP对象 //需MSXML4.0支持2.打开与服务端的连接,同时定义指令发送方式,服务网页(URL)和请求权限等.客户端通过Open命令打开与服务端的服务网页的连接.与 ...
- TypeScript `unknown` 类型
unknown 字面理解和 any 其实没差,任何类型都可赋值给它,但有一点, Anything is assignable to unknown, but unknown isn't assigna ...
- Set-----集合入门
函数中的集合和 数学中的集合 基本上差不多 集合中每个元素最多只能出现一次 并且 当元素储存到set集合之中 会自动 按照 ascll 进行 从小到大的 排序 大神关于 set 的 详 ...
- ansible基础知识(二)
软件相关模块 yum yum和rpm的区别 rpm: (Redhat package manager)RPM管理支持事务机制.增强了程序安装卸载的管理. yum: YUM被称为 Yellow dog ...