3624: [Apio2008]免费道路

Time Limit: 2 Sec  Memory Limit: 128 MBSec  Special Judge
Submit: 1292  Solved: 518
[Submit][Status][Discuss]

Description

Input

Output

Sample Input

5 7 2
1 3 0
4 5 1
3 2 0
5 3 1
4 3 0
1 2 1
4 2 1

Sample Output

3 2 0
4 3 0
5 3 1
1 2 1

HINT

 

Source

自己的第一次思路:
  小数据枚举k条鹅卵石路,大数据随机找k条鹅卵石路,跟剩下的水泥路构树,如果可以构成树,则此方案可行。
  随机化的阈值我设置的是20.

  然后就砍到72分。

#include<cmath>
#include<ctime>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int N=2e4+;
const int M=1e5+;
struct edge{int u,v,w,id;}e[M],z[M];
int n,m,num0,K,tot,cct,fa[N],ans[M];
int a[];bool vis[];
inline int read(){
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
}
inline bool cmp(const edge &a,const edge &b){
return a.id<b.id;
}
inline bool cmp2(const edge &a,const edge &b){
return a.w<b.w;
}
int find(int x){
return fa[x]==x?x:fa[x]=find(fa[x]);
}
inline void pre(){
tot=;ans[]=;
for(int i=;i<=n;i++) fa[i]=i;
random_shuffle(e+,e+num0+);
for(int i=,x,y;i<=num0;i++){
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;ans[++ans[]]=i;
if(++tot==K) break;
}
}
}
inline void ord(){
tot=;ans[]=;
for(int i=;i<=n;i++) fa[i]=i;
for(int i=,x,y;i<=K;i++){
x=find(e[a[i]].u);y=find(e[a[i]].v);
if(x!=y){
fa[y]=x;ans[++ans[]]=a[i];
if(++tot==K) break;
}
}
}
inline void work(){
for(int i=num0+,x,y;i<=m;i++){
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;ans[++ans[]]=i;
if(++tot==n-) break;
}
}
}
inline void print(){
for(int i=;i<=ans[];i++) z[i]=e[ans[i]];
sort(z+,z+ans[]+,cmp);
for(int i=;i<=ans[];i++) printf("%d %d %d\n",z[i].u,z[i].v,z[i].w);
}
void dfs(int x){
if(x>K){
ord();work();
if(tot==n-){print();exit();}
return ;
}
for(int i=a[x-]+;i<=num0;i++){
if(!vis[i]){
vis[i]=;
a[x]=i;
if(num0-i<K-x) break;
dfs(x+);
vis[i]=;
a[x]=;
}
}
}
int main(){
// freopen("sh.txt","r",stdin);
srand(time());
n=read();m=read();K=read();
for(int i=;i<=m;i++){
e[i].u=read(),e[i].v=read(),e[i].w=read(),e[i].id=i;
if(!e[i].w) num0++;
}
sort(e+,e+m+,cmp2);
if(K<=){dfs();puts("no solution");return ;}
while(){
pre();
if(tot!=n-) work();
if(tot==n-){print();return ;}
if(++cct==) break;
}
puts("no solution");
// cnt=ans[0];ans[0]=0;
// for(int i=1;i<=ans[0];i++) printf("%d %d %d\n",e[ans[i]].u,e[ans[i]].v,e[ans[i]].w);
/*for(int i=1;i<=cnt;i++) z[i]=e[ans[i]];
sort(z+1,z+cnt+1,cmp2);
for(int i=1;i<=n;i++) fa[i]=i;
for(int i=1,tot=0,x,y;i<=cnt;i++){
x=find(z[i].u);y=find(z[i].v);
if(x!=y){
fa[y]=x;ans[++ans[0]]=i;
if(++tot==n-1) break;
}
}*/
/*cnt=ans[0];ans[0]=0;
for(int i=1;i<=cnt;i++) e[i]=z[ans[i]];
sort(e+1,e+cnt+1,cmp);*/
// for(int i=1;i<=cnt;i++) printf("%d %d %d\n",e[i].u,e[i].v,e[i].w);
return ;
}

代码留念

自己的第二次思路:(后悔当时为什么没有多想想)
  2次kruscal解决。
  第一次kruscal:首先考虑把所有水泥路连上。如果构不成树,则需要用鹅卵石路填边,这些鹅卵石路是必须要的鹅卵石路(如果必须要的鹅卵石路的数量>K直接无解);剩下的鹅卵石路都是不必要的。
  第二次kruscal:先把上一次找到的必须要的鹅卵石路填上,此时的鹅卵石路不一定恰有K条,可能比K条少,于是考虑优先连不必要的鹅卵石路,直到凑满K条为止,然后剩下的边随便找几条水泥路连起来就好(special judge告诉我们输出任意解均可)

ps:

  边权只有01的图,生成树的权值和可以取到任意的介于[MST,MBT]的任意值,其中MST表示最小生成树,MBT最大。

  我们可以发现MST和MBT的区别在与其中一些点,这些点与生成树联通的边可以选择0或者1,所以你可以把一些点的边替换,每次权值变化1,所以可以取到任意的权值.

#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
const int N=4e4+;
const int M=2e5+;
struct edge{int u,v,w,tag;}e[M];
int n,m,K,tot,fa[N];
inline int read(){
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
}
int find(int x){
return fa[x]==x?x:fa[x]=find(fa[x]);
}
inline void ReadData(){
n=read();m=read();K=read();
for(int i=;i<=m;i++) e[i].u=read(),e[i].v=read(),e[i].w=read(),e[i].tag=;
}
inline void Kruscal1(){
int cct=;
for(int i=;i<=n;i++) fa[i]=i;
for(int i=,x,y;i<=m;i++) if(e[i].w){//先铺水泥路
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;
if(++cct==n-) break;
}
}
if(cct==n-) return ;
for(int i=,x,y;i<=m;i++) if(!e[i].w){
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;tot++;e[i].tag=;//必要鹅卵石
if(++cct==n-) break;
}
}
}
inline void Kruscal2(){
int cct=;
for(int i=;i<=n;i++) fa[i]=i;
for(int i=,x,y;i<=m;i++) if(e[i].tag){//整理找出的必要鹅卵石
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;
if(++cct==n-) break;
}
}
for(int i=,x,y;i<=m;i++) if(!e[i].w){
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;++cct;e[i].tag=;//不必要鹅卵石补齐K条
if(++tot==K) break;
}
}
if(tot!=K){puts("no solution");exit();}
for(int i=,x,y;i<=m;i++) if(e[i].w){//水泥路补齐n-1条
x=find(e[i].u);y=find(e[i].v);
if(x!=y){
fa[y]=x;e[i].tag=;
if(++cct==n-) break;
}
}
}
inline void WriteAns(){
for(int i=;i<=m;i++) if(e[i].tag) printf("%d %d %d\n",e[i].u,e[i].v,e[i].w);
}
int main(){
ReadData();
Kruscal1();
Kruscal2();
WriteAns();
return ;
}

[Apio2008]免费道路[Kruscal]的更多相关文章

  1. 题解 Luogu P3623 [APIO2008]免费道路

    [APIO2008]免费道路 题目描述 新亚(New Asia)王国有 N 个村庄,由 M 条道路连接.其中一些道路是鹅卵石路,而其它道路是水泥路.保持道路免费运行需要一大笔费用,并且看上去 王国不可 ...

  2. [BZOJ3624][Apio2008]免费道路

    [BZOJ3624][Apio2008]免费道路 试题描述 输入 输出 输入示例 输出示例 数据规模及约定 见“输入”. 题解 第一步,先尽量加入 c = 1 的边,若未形成一个连通块,则得到必须加入 ...

  3. bzoj 3624: [Apio2008]免费道路 生成树的构造

    3624: [Apio2008]免费道路 Time Limit: 2 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 111  Solved: 4 ...

  4. BZOJ 3624: [Apio2008]免费道路

    3624: [Apio2008]免费道路 Time Limit: 2 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 1201  Solved:  ...

  5. P3623 [APIO2008]免费道路

    3624: [Apio2008]免费道路 Time Limit: 2 Sec Memory Limit: 128 MBSec Special Judge Submit: 2143 Solved: 88 ...

  6. Kruskal算法及其类似原理的应用——【BZOJ 3654】tree&&【BZOJ 3624】[Apio2008]免费道路

    首先让我们来介绍Krukal算法,他是一种用来求解最小生成树问题的算法,首先把边按边权排序,然后贪心得从最小开始往大里取,只要那个边的两端点暂时还没有在一个联通块里,我们就把他相连,只要这个图里存在最 ...

  7. [APIO2008]免费道路

    [APIO2008]免费道路 BZOJ luogu 先把必须连的鹅卵石路连上,大于k条no solution 什么样的鹅卵石路(u,v)必须连?所有水泥路都连上仍然不能使u,v连通的必须连 补全到k条 ...

  8. [APIO2008]免费道路(生成树)

    新亚(New Asia)王国有 N 个村庄,由 M 条道路连接.其中一些道路是鹅卵石路,而其它道路是水泥路.保持道路免费运行需要一大笔费用,并且看上去 王国不可能保持所有道路免费.为此亟待制定一个新的 ...

  9. 【bzoj3624】Apio2008—免费道路

    http://www.lydsy.com/JudgeOnline/problem.php?id=3624 (题目链接) 题意 给出一张无向图,其中有0类边和1类边.问能否构成正好有K条0类边的生成树, ...

随机推荐

  1. c fopen fread 错误

    真的被,读取一个txt文本,结果一个早上都没搞好 程序如下: 能看出哪里有问题么,输出字符串,得到的结果后面有“屯”或则 “烫”,单个字符输出来也有,为何,搜啊搜,改txt的内容,依旧不行 最后 改f ...

  2. Memcached存储机制

    Memcached存储机制 memcached 内存管理 分析(转) 缓存.缓存算法和缓存框架简介 memcached全面剖析–PDF总结篇

  3. mysql Communication link failure, message from server: "Can't get hostname for your address"

    在连接mysql jdbc时候,抛出了 com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: Communicat ...

  4. C++入门学习

    使用cout和cin,需要包括C++提供的输入输出库.#include <iostream>    标准输入输出头文件 <string>        using namesp ...

  5. 元素随屏幕滚动到顶部固定js效果

    网站中常见这种效果,某个广告或详情页切换tab,当屏幕向下移动时,该元素会停留在浏览器最顶部,下面ecshop模板中心教您实现js代码: 案例图: 1.首先在页面上找到该元素  加上 id =&quo ...

  6. windows2008 使用nginx 反向代理实现负载均衡解决HTTPS 证书问题

    由于项目需要 负载均衡由NBL 转成nginx 反向代理.考虑都是https模块,所以证书成了个难题. 解决方案: 1.下载openssl(windows 安装包) 2.打开bin/下面的openss ...

  7. docker dockerfile构建自己的tomcat镜像

    文件 1.apache-tomcat-8.5.24.tar.gz, jdk-8u151-linux-x64.tar.gz Dockerfile文件: FROM centosWORKDIR /usrCO ...

  8. python带参装饰器的改良版

    简单点就是这种 def deco2(param=1): def _deco2(fun): def __deco2(*args, **kwargs): print (param) fun(*args, ...

  9. 使用Matplotlib画图系列(一)

    实现一个最简单的plot函数调用: import matplotlib.pyplot as plt y=pp.DS.Transac_open # 设置y轴数据,以数组形式提供 x=len(y) # 设 ...

  10. html+jquery制作网页地图

    http://jvectormap.com/ <!--StartFragment --> JVectorMap 是一个显示矢量地图的jQuery插件.它使用 SVG 在Firefox 3 ...