前言:

如果我自己写的话,或许能想出来正解,但是多半会因为整不出正确性而弃掉。

解析:

这题算是对Kruskal的熟练运用吧。

要求一颗生成树。也就是说,最后的边数是确定的。

首先我们容易想到一个策略:

先跑Kruskal,优先选k条石子路,剩下的选水泥路。

但是这样做显然是错误的。

因为,当随便选了k条石子路后,可能出现:

发现无论怎么选(n-1-k)条水泥路,也无法使图连通。如果这时选一条石子路,就可以保证连通性。

但是,发现这时已经选满了k条石子路,就没法再选石子路了。

我们可以通过设计一个策略来使这种情况不发生。

首先,我们可以优先用水泥路跑Kruskal,这时如果有一条石子路,如果不加上它,就无法保证连通性,那就将其打上标记,意思是这条石子路一定会出现在最后的答案里面。

第二遍Kruskal,先将上次打过标记的石子路加入并查集。然后,将石子路补成k条。

第三遍,补上水泥路。

这个策略为什么是正确的呢?

首先,可以发现,(设第一遍Kruskal找出了cnt条石子路)

执行第一遍Kruskal后,假如存在生成树,那么我们必然可以通过cnt条石子路+一些水泥路的方式来找到一颗生成树。

那么,此时我们再选一些石子路,相当于把一些水泥路换成了石子路。假如原来是有方案的,那么后面一定是有方案的。

代码:
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxm=100000+10,maxn=20000+10;
#define gc() (p1 == p2 ? (p2 = buf + fread(p1 = buf, 1, 1 << 20, stdin), p1 == p2 ? EOF : *p1++) : *p1++)
#define read() ({ register int x = 0, f = 1; register char c = gc(); while(c < '0' || c > '9') { if (c == '-') f = -1; c = gc();} while(c >= '0' && c <= '9') x = x * 10 + (c & 15), c = gc(); f * x; })
char buf[1 << 20], *p1, *p2;
struct node{
int from,to,op,flag;
}b[maxm],ans[maxm];
int n,m,k,cnt,tot;
int fa[maxn];
bool cmp1(node x,node y){
return x.op>y.op;
}
int find(int x){
return fa[x]==x ? x : (fa[x]=find(fa[x])) ;
}
void Merge(int x,int y){
int rx=find(x);
int ry=find(y);
if(rx==ry) return;
fa[rx]=ry;
}
void K1(){
for(int i=1;i<=n;++i) fa[i]=i;
for(int i=1;i<=m;++i){
int x=b[i].from;
int y=b[i].to;
int rx=find(x);
int ry=find(y);
if(rx==ry) continue;
fa[rx]=ry;
if(b[i].op==0){
b[i].flag=1;
cnt++;
}
}
}
void K2(){
for(int i=1;i<=n;++i) fa[i]=i;
for(int i=1;i<=m;++i) if(b[i].flag) Merge(b[i].from,b[i].to);
for(int i=1;i<=m;++i){
if(b[i].op) continue;
int x=b[i].from;
int y=b[i].to;
int rx=find(x);
int ry=find(y);
if(rx==ry) continue;
fa[rx]=ry;
b[i].flag=1;
cnt++;
if(cnt==k) break;
}
}
void K3(){
for(int i=1;i<=m;++i){
if(b[i].op==0) continue;
int x=b[i].from;
int y=b[i].to;
int rx=find(x);
int ry=find(y);
if(rx==ry) continue;
fa[rx]=ry;
b[i].flag=1;
}
}
void Solve(){
scanf("%d%d%d",&n,&m,&k);
for(int i=1;i<=m;++i) scanf("%d%d%d",&b[i].from,&b[i].to,&b[i].op);
sort(b+1,b+m+1,cmp1);
K1();
if(cnt>k){
printf("-1\n");
return;
}
int x=find(1);
for(int i=2;i<=n;++i){
if(find(i)!=x) {
printf("-1\n");
return;
}
}
K2();
if(cnt<k){
printf("-1\n");
return;
}
K3();
for(int i=1;i<=m;++i){
if(b[i].flag){
printf("%d %d %d\n",b[i].from,b[i].to,b[i].op);
}
}
}
int main(){
Solve();
return 0;
}

[火星补锅] 水题大战Vol.2 T2 && luogu P3623 [APIO2008]免费道路 题解的更多相关文章

  1. [火星补锅] 水题大战Vol.2 T1 && luogu P1904 天际线 题解 (线段树)

    前言: 当时考场上并没有想出来...后来也是看了题解才明白 解析: 大家(除了我)都知道,奇点和偶点会成对出现,而出现的前提就是建筑的高度突然发生变化.(这个性质挺重要的,我之前没看出来) 所以就可以 ...

  2. 水题大战Vol.3 B. DP搬运工2

    水题大战Vol.3 B. DP搬运工2 题目描述 给你\(n,K\),求有多少个\(1\)到\(n\) 的排列,恰好有\(K\)个数\(i\) 满足\(a_{i-1},a_{i+1}\) 都小于\(a ...

  3. 金题大战Vol.0 A、凉宫春日的叹息

    金题大战Vol.0 A.凉宫春日的叹息 题目描述 给定一个数组,将其所有子区间的和从小到大排序,求第 \(k\) 小的是多少. 输入格式 第一行两个数\(n\),$ k\(,表示数组的长度和\)k$: ...

  4. 金题大战Vol.0 B、序列

    金题大战Vol.0 B.序列 题目描述 给定两个长度为 \(n\) 的序列\(a\), \(b\). 你需要选择一个区间\([l,r]\),使得\(a_l+-+a_r>=0\)且\(b_l+-+ ...

  5. 金题大战Vol.0 C、树上的等差数列

    金题大战Vol.0 C.树上的等差数列 题目描述 给定一棵包含\(N\)个节点的无根树,节点编号\(1-N\).其中每个节点都具有一个权值,第\(i\)个节点的权值是\(A_i\). 小\(Hi\)希 ...

  6. 土题大战Vol.0 A. 笨小猴 思维好题

    土题大战Vol.0 A. 笨小猴 思维好题 题目描述 驴蛋蛋有 \(2n + 1\) 张 \(4\) 星武器卡片,每张卡片上都有两个数字,第 \(i\) 张卡片上的两个数字分别是 \(A_i\) 与 ...

  7. 火题大战Vol.1 A.

    火题大战Vol.1 A. 题目描述 给定两个数\(x\),\(y\),比较\(x^y\) 与\(y!\)的大小. 输入格式 第一行一个整数\(T\)表示数据组数. 接下来\(T\)行,每行两个整数\( ...

  8. 火题大战Vol.0 B 计数DP

    火题大战Vol.0 B 题目描述 \(n\) 个沙茶,被编号 \(1\)~$ n$.排完队之后,每个沙茶希望,自己的相邻的两人只要无一个人的编号和自己的编号相差为 \(1\)(\(+1\) 或\(-1 ...

  9. [火星补锅] 非确定性有穷状态决策自动机练习题Vol.3 T3 && luogu P4211 [LNOI2014]LCA 题解

    前言: 这题感觉还是很有意思.离线思路很奇妙.可能和二次离线有那么一点点相似?当然我不会二次离线我就不云了. 解析: 题目十分清真. 求一段连续区间内的所有点和某个给出的点的Lca的深度和. 首先可以 ...

随机推荐

  1. SpingBoot-Dubbo-Zookeeper-分布式

    目录 分布式理论 什么是分布式系统? Dubbo文档 单一应用架构 垂直应用架构 分布式服务架构 流动计算架构 什么是RPC RPC基本原理 测试环境搭建 Dubbo Dubbo环境搭建 Window ...

  2. STP(生成树)的概述和工作原理

    一.STP的简介 二.STP的概述 三.STP的工作原理 四.生成树的算法 五.BPDU(桥协议数据单元) 六.STP收敛 TP简介 STP - Spanning Tree Protocol(生成树协 ...

  3. 六种多线程方法解决UI线程堵塞

    http://blog.csdn.net/oyi319/article/details/6851371 一.六种多线程方法 .NET Framework2.0框架提供了至少4种方式实现多线程,它们是& ...

  4. Php实现简易购物商城系统

    实现功能: 1.系统功能模块包括: 1)登陆注册模块 包括验证码.找回密码.注册模块中要使用Ajax判断用户名是否已经存在,使用正则表达式判断电子邮件.手机号和用户密码的格式是否合法. 2)用户管理模 ...

  5. java.lang.SecurityException: MODE_WORLD_READABLE错误解决

    问题描述:运行Android项目有以下报错: 解决方法: 把 MODE_WORLD_READABLE 更换成 MODE_PRIVATE 即可,因为MODE_WORLD_READABLE 模式已经被废弃 ...

  6. Django学习day03随堂笔记

    每日测验 """ 今日考题 1.什么是静态文件,django静态文件配置如何配置,如何解决接口前缀不断变化,html页面上路径的引用需要反复修改的问题 2.request ...

  7. 利用 uber-go/dig 库管理依赖

    利用 uber-go/dig 库管理依赖 github 地址 官方文档 介绍 dig 库是一个为 go 提供依赖注入 (dependency injection) 的工具包,基于 reflection ...

  8. 动态查看及加载PHP扩展

    在编译并完成 php.ini 的配置之后,我们就成功的安装了一个 PHP 的扩展.不过, PHP 也为我们提供了两个在动态运行期间可以查看扩展状态以及加载未在 php.ini 中进行配置的扩展的函数. ...

  9. Java基础(六)——集合

    一.概述 1.介绍 为什么出现集合? 答:面向对象语言对事物的体现都是以对象的形式,所以为了方便对多个对象的操作,对对象进行存储,集合就是存储对象最常用的一种方式. 数组和集合类同是容器,有何不同? ...

  10. 搭建http文件服务器 - python3使用http.server搭建http文件服务器

    适用场景 只要有python3就可以,windows系统cmd窗口直接敲命令,Linux系统,直接敲命令 step-1 cd cd 到需要搭建服务器的目录 step-2 http.server pyt ...