【Kruskal+贪心思想】BZOJ3624-[Apio2008]免费道路
国庆万岁!!!!!
【题目大意】
给定一张无向图,有两种边的类型为0和1。求一个最小生成树使得边0有k条。
【思路】
跑两次Kruskal。
第一次的时候优先选择边1,然后判断有哪些边0还不能连通,那么这些边0是必须要选取的。如果必须要选的边0大于k,那么直接输出无解。
第二次的时候先合并那么必须要选取的边0,然后在剩下的边0中左右还没有连通的里选取。如果把所有都选上了之后边0的数量还是没有到k,那么直接输出无解。
截下来按照普通Kruskal的方法把边1合并掉。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
const int MAXN=+;
const int MAXM=+;
struct Edge
{
int u,v;
}edge[MAXM];
int n,m,k;
int vis[MAXM];
int L,R;
int fa[MAXN],h[MAXN];
vector<int> mustchoose; void initset(){for (int i=;i<=n;i++) fa[i]=i,h[i]=;} int find(int x)
{
int r=x;
while (fa[r]!=r) r=fa[r];
while (fa[x]!=r)
{
int tmp=fa[x];
fa[x]=r;
x=fa[x];
}
return r;
} void unionset(int a,int b)
{
if (h[a]>=h[b])
{
fa[b]=a;
if (h[a]==h[b]) h[a]++;
}
else fa[a]=b;
} void init()
{
scanf("%d%d%d",&n,&m,&k);
L=,R=m+;
for (int i=;i<=m;i++)
{
int u,v,c;
scanf("%d%d%d",&u,&v,&c);
if (c) edge[++L]=(Edge){u,v};
else edge[--R]=(Edge){u,v};
}
} void solve()
{
int t=;
initset();
for (int i=;i<=m;i++)
{
int fa=find(edge[i].u),fb=find(edge[i].v);
if (fa!=fb)
{
unionset(fa,fb);
if (i>=R)
{
vis[i]=;
t++;
mustchoose.push_back(i);
}
}
}
if (t>k)
{
puts("no solution");
return;
}
// 找出必须要选择的鹅卵石路 initset();
for (int i=;i<mustchoose.size();i++)
{
int fa=find(edge[mustchoose[i]].u),fb=find(edge[mustchoose[i]].v);
unionset(fa,fb);
}
for (int i=R;i<=m;i++)
if (t<k && !vis[i])
{
int fa=find(edge[i].u),fb=find(edge[i].v);
if (fa!=fb)
{
unionset(fa,fb);
vis[i]=;
t++;
}
}
if (t<k)
{
puts("no solution");
return;
}
//先选择必须要的鹅卵石路,然后再用其他鹅卵石路填充 for (int i=;i<=L;i++)
{
int fa=find(edge[i].u),fb=find(edge[i].v);
if (fa!=fb)
{
unionset(fa,fb);
vis[i]=;
}
}
for (int i=;i<=L;i++) if (vis[i]) printf("%d %d %d\n",edge[i].u,edge[i].v,);
for (int i=R;i<=m;i++) if (vis[i]) printf("%d %d %d\n",edge[i].u,edge[i].v,);
} int main()
{
init();
solve();
}
【Kruskal+贪心思想】BZOJ3624-[Apio2008]免费道路的更多相关文章
- [BZOJ3624][Apio2008]免费道路
[BZOJ3624][Apio2008]免费道路 试题描述 输入 输出 输入示例 输出示例 数据规模及约定 见“输入”. 题解 第一步,先尽量加入 c = 1 的边,若未形成一个连通块,则得到必须加入 ...
- BZOJ3624: [Apio2008]免费道路(最小生成树)
题意 题目链接 Sol 首先答案一定是一棵树 这棵树上有一些0边是必须要选的,我们先把他们找出来,如果数量$\geqslant k$显然无解 再考虑继续往里面加0的边,判断能否加到k条即可 具体做法是 ...
- Kruskal算法及其类似原理的应用——【BZOJ 3654】tree&&【BZOJ 3624】[Apio2008]免费道路
首先让我们来介绍Krukal算法,他是一种用来求解最小生成树问题的算法,首先把边按边权排序,然后贪心得从最小开始往大里取,只要那个边的两端点暂时还没有在一个联通块里,我们就把他相连,只要这个图里存在最 ...
- BZOJ 3624: [Apio2008]免费道路
3624: [Apio2008]免费道路 Time Limit: 2 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 1201 Solved: ...
- bzoj 3624: [Apio2008]免费道路 生成树的构造
3624: [Apio2008]免费道路 Time Limit: 2 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 111 Solved: 4 ...
- 题解 Luogu P3623 [APIO2008]免费道路
[APIO2008]免费道路 题目描述 新亚(New Asia)王国有 N 个村庄,由 M 条道路连接.其中一些道路是鹅卵石路,而其它道路是水泥路.保持道路免费运行需要一大笔费用,并且看上去 王国不可 ...
- [Apio2008]免费道路[Kruscal]
3624: [Apio2008]免费道路 Time Limit: 2 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 1292 Solved: ...
- P3623 [APIO2008]免费道路
3624: [Apio2008]免费道路 Time Limit: 2 Sec Memory Limit: 128 MBSec Special Judge Submit: 2143 Solved: 88 ...
- [APIO2008]免费道路
[APIO2008]免费道路 BZOJ luogu 先把必须连的鹅卵石路连上,大于k条no solution 什么样的鹅卵石路(u,v)必须连?所有水泥路都连上仍然不能使u,v连通的必须连 补全到k条 ...
随机推荐
- 【洛谷P1597】语句解析
题目背景 木有背景…… 题目描述 一串(<255)PASCAL语言,只有a,b,c 3个变量,而且只有赋值语句,赋值只能是一个一位的数字或一个变量,未赋值的变量值为0.输出a,b,c 最终的值. ...
- Metasploit 进阶
本文是"T00LS Metasploit(第二季)"的文档版,是个人在观看视频动手操作的一个记录,仅供学习.文中会介绍Metasploit的一些基本使用:主要包括远程代码执行.MI ...
- 24、CSS定位
CSS定位方法 driver.find_element_by_css_selector() 1.CSS定位常用策略(方式) 1.id选择器 说明:根据元素id属性来选择 格式:#id 如:#userA ...
- koa源码阅读[3]-koa-send与它的衍生(static)
koa源码阅读的第四篇,涉及到向接口请求方提供文件数据. 第一篇:koa源码阅读-0第二篇:koa源码阅读-1-koa与koa-compose第三篇:koa源码阅读-2-koa-router 处理静态 ...
- wordpress 模板制作之一
WP模板工作原理图:
- Installation Guide for Appium 1.6.3
A.) System Requirements : - Require node 4 or above Xcode 8 iOS 10 B.) Open terminal and type follow ...
- Struts结果跳转方式(四种result配置)
1.转发(默认转发)
- 选择问题(选择数组中第K小的数)
由排序问题可以引申出选择问题,选择问题就是选择并返回数组中第k小的数,如果把数组全部排好序,在返回第k小的数,也能正确返回,但是这无疑做了很多无用功,由上篇博客中提到的快速排序,稍稍修改下就可以以较小 ...
- 离线部署ELK+kafka日志管理系统【转】
转自 离线部署ELK+kafka日志管理系统 - xiaoxiaozhou - 51CTO技术博客http://xiaoxiaozhou.blog.51cto.com/4681537/1854684 ...
- php PDO判断连接是否可用的方法
转载自:傲雪星枫 原文地址: http://blog.csdn.net/fdipzone/article/details/53117541 mysql_ping() 检查到服务器的连接是否正常.如果 ...