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. php-fpm的pool php-fpm慢执行日志 open_basedir php-fpm进程管理

    php-fpm的pool • vim /usr/local/php/etc/php-fpm.conf//在[global]部分增加 • include = etc/php-fpm.d/*.conf • ...

  2. 简单理解IoC与DI

    为了理解Spring的IoC与DI从网上查了很多资料,作为初学者,下面的描述应该是最详细,最易理解的方式了. 首先想说说IoC(Inversion of Control,控制倒转).这是spring的 ...

  3. PowerDesigner使用技巧(转载)

    1.如何打开PowerDesigner 快捷工具栏 paletteTools(工具栏)--> customsize toolbars(自定义工具栏)-->勾选 palette(调色板) 2 ...

  4. SharePoint 2013 显示“以其他用户身份登录”菜单项

    最近在SharePoint 2013的网站上发现,没有看到有切换不同用户登录的入口,在SharePoint 2010中是存在这样的菜单项能够很方便的进行用户切换的,不知道为什么,SharePoint ...

  5. PS1-4

    export PS2="continue->" cat ps4.sh export PS4='$0.$LINENO+ ' set -x echo "PS4 demo ...

  6. WebGL 颜色与纹理

    1.纹理坐标 纹理坐标是纹理图像上的坐标,通过纹理坐标可以在纹理图像上获取纹理颜色.WebGL系统中的纹理坐标系统是二维的,如图所示.为了将纹理坐标和广泛使用的x.y坐标区分开来,WebGL使用s和t ...

  7. ESPCN处理彩色图像代码

    原来仅处理了Y通道,输出的灰度图像. Super-Resolution/ESPCN at master · wangxuewen99/Super-Resolution · GitHub https:/ ...

  8. mysql 两台主主复制配置

    A.服务器 [mysqld] # The TCP/IP Port the MySQL Server will listen on port= server-id= #master-host=10.1. ...

  9. Tensorflow同时加载使用多个模型

    在Tensorflow中,所有操作对象都包装到相应的Session中的,所以想要使用不同的模型就需要将这些模型加载到不同的Session中并在使用的时候申明是哪个Session,从而避免由于Sessi ...

  10. 利用ROS工具从bag文件中提取图片

    bag文件是ROS常用的数据存储格式,因此要从bag文件中提取数据就需要了解一点ROS的背景知识. 1. 什么是ROS及其优势 ROS全称Robot Operating System,是BSD-lic ...