F2 - Spanning Tree with One Fixed Degree - 并查集+DFS
这道题还是非常有意思的,题意很简单,就是给定一个图,和图上的双向边,要求1号节点的度(连接边的条数)等于K,求这棵树的生成树。
我们首先要解决,如何让1号节点的度时为k的呢???而且求的是生成树,意思是不是所有边都会选择。那么我们如何选择才能保证1号节点有K个度呢???这里就要考虑联通分量的问题了,我们刨除1号点,那么联通分量的个数,就是我们让图联通的最小个数,因此我们需要用并查集,把点分在不同的联通块内部。
再考虑我们每个联通块,至少需要1条连接1号点的边。不够K再添加连接1号点的边。
然后考虑由于是生成树,我们可以枚举每个联通块连接1的点,DFS找出生成树边,记录即可。
DFS+并查集版本:
#include<iostream>
#include<string.h>
#include<stdio.h>
#include<algorithm>
#include<vector>
#define pii pair<int,int>
#define mp make_pair
using namespace std;
const int maxx = 2e5+;
int fa[maxx];
vector<int>G[maxx];
vector<int>p;
vector<pii>ans;
struct node
{
int u,v;
} a[maxx];
int vis[maxx];
int v[maxx];
int Find(int x)
{
return fa[x]==x?x:(fa[x]=Find(fa[x]));
}
void dfs(int x)
{
int nex;
for (int i=; i<G[x].size(); i++)
{
nex=G[x][i];
if (nex==)continue;
if (v[nex]==)
{
v[nex]=;
ans.push_back(mp(x,nex));
dfs(nex);
}
}
}
int main()
{
int n,m,k;
while(~scanf("%d%d%d",&n,&m,&k))
{
ans.clear();
memset(vis,,sizeof(vis));
memset(v,,sizeof(v));
for (int i=; i<=n; i++)
{
fa[i]=i;
}
for (int i=; i<=m; i++)
{
scanf("%d%d",&a[i].u,&a[i].v);
G[a[i].u].push_back(a[i].v);
G[a[i].v].push_back(a[i].u);
if (a[i].u!= && a[i].v!=)
{
int fx=Find(a[i].u),fy=Find(a[i].v);
fa[fx]=fy;
}
}
int cnt=;
for (int i=; i<=m; i++)
{
if (a[i].u== || a[i].v==)
{
int fx=Find(a[i].u),fy=Find(a[i].v);
if (fx!=fy)
{
vis[i]=;
if (a[i].u==)
{
p.push_back(a[i].v);
}
else
{
p.push_back(a[i].u);
}
v[a[i].u]=;
v[a[i].v]=;
ans.push_back(mp(a[i].u,a[i].v));
cnt++;
fa[fx]=fy;
}
else
{
vis[i]=;
}
}
}
if (cnt>k)
{
printf("NO\n");
continue;
}
k-=cnt;
for (int i=; i<=m; i++)
{
if (k==)break;
if(vis[i]==)
{
k--;
ans.push_back(mp(a[i].u,a[i].v));
if (a[i].u==){
p.push_back(a[i].v);
}else {
p.push_back(a[i].u);
}
v[a[i].u]=;
v[a[i].v]=;
vis[i]=;
}
}
if (k!=)
{
printf("NO\n");
continue;
}
v[]=;
printf("YES\n");
for (int i=; i<p.size(); i++)
{
dfs(p[i]);
}
for (int i=; i<ans.size(); i++)
{
printf("%d %d\n",ans[i].first,ans[i].second);
}
}
return ;
}
当然有大佬提出了更牛逼的做法,仍然是用并查集,单独处理连接1的边,然后枚举每条边,当这个点的不是指向1的,并且边的两点却不在一个连通分量里面,那么这条边是必选的。
并查集版本:
#include<iostream>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<vector>
using namespace std;
const int maxx = 2e5+;
struct node
{
int u,v;
} a[maxx];
int fa[maxx];
int vis[maxx];
int Find(int x)
{
return fa[x]==x?x:(fa[x]=Find(fa[x]));
};
int add(int x,int y)
{
int fx=Find(x),fy=Find(y);
if(fx!=fy)fa[fx]=fy;
}
int main()
{
int n,m,d,k;
int u,v;
while(~scanf("%d%d%d",&n,&m,&k))
{
memset(vis,,sizeof(vis));
for (int i=; i<=n; i++)
{
fa[i]=i;
}
for (int i=; i<=m; i++)
{
scanf("%d%d",&a[i].u,&a[i].v);
if (a[i].u!= && a[i].v!=)
{
int fx=Find(a[i].u),fy=Find(a[i].v);
if (fx!=Find(fy))
{
fa[fx]=fy;
}
}
}
int cnt=;
for (int i=; i<=m; i++)
{
if (a[i].u== || a[i].v==)
{
int fx=Find(a[i].u),fy=Find(a[i].v);
if(fx!=fy)
{
vis[i]=;//必选
fa[fx]=fy;
cnt++;
}
else
{
vis[i]=;//备选
}
}
}
if (cnt>k)
{
printf("NO\n");
continue;
}
k-=cnt;
for (int i=; i<=m; i++) //选择剩下的和1相连的数目
{
if (k==)break;
if (vis[i]==)
{
vis[i]=;
//cout<<i<<endl;
k--;
int fx=Find(a[i].u),fy=Find(a[i].v);
fa[fx]=fy;
}
}
if (k!=)
{
printf("NO\n");
continue;
}
for (int i=; i<=n; i++) //再次初始化
{
fa[i]=i;
}
for (int i=;i<=m;i++){
if (vis[i]==){
int fx=Find(a[i].u),fy=Find(a[i].v);
fa[fx]=fy;
}
}
for (int i=; i<=m; i++)
{
if (a[i].u!=a[i].v && a[i].u!= && a[i].v!=)
{
int fx=Find(a[i].u),fy=Find(a[i].v);
if (fx!=fy)
{
fa[fx]=fy;
vis[i]=;
}
}
}
printf("YES\n");
for (int i=;i<=m;i++){
if(vis[i]==) printf("%d %d\n",a[i].u,a[i].v);
}
}
return ;
}
F2 - Spanning Tree with One Fixed Degree - 并查集+DFS的更多相关文章
- Codeforces 1133 F2. Spanning Tree with One Fixed Degree 并查集+生成树
好久没更新博客了,一直懒得动,这次更新一下. 题意大概是:给出一个图,求它的一个一号节点的度数恰好为D的生成树的方案. 一开始随便水了个乱搞贪心,不出意外并没有过. 仔细思考之后,对于这个问题我们可以 ...
- UVALive 6910 Cutting Tree(离线逆序并查集)
[题目]:(地址:) http://acm.hust.edu.cn/vjudge/contest/view.action?cid=97671#problem/E [题意]: 给出多棵树和两类操作:操作 ...
- 第46届ICPC澳门站 K - Link-Cut Tree // 贪心 + 并查集 + DFS
原题链接:K-Link-Cut Tree_第46屆ICPC 東亞洲區域賽(澳門)(正式賽) (nowcoder.com) 题意: 要求一个边权值总和最小的环,并从小到大输出边权值(2的次幂):若不存在 ...
- HDU 4582 DFS spanning tree(DFS+贪心)(2013ACM-ICPC杭州赛区全国邀请赛)
Problem Description Consider a Depth-First-Search(DFS) spanning tree T of a undirected connected gra ...
- Codeforces Educational Codeforces Round 3 E. Minimum spanning tree for each edge LCA链上最大值
E. Minimum spanning tree for each edge 题目连接: http://www.codeforces.com/contest/609/problem/E Descrip ...
- BNUOJ 26229 Red/Blue Spanning Tree
Red/Blue Spanning Tree Time Limit: 2000ms Memory Limit: 131072KB This problem will be judged on HDU. ...
- 多校 HDU - 6614 AND Minimum Spanning Tree (二进制)
传送门 AND Minimum Spanning Tree Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 ...
- E - Minimum Spanning Tree Gym - 102220E (转化+贡献)
In the mathematical discipline of graph theory, the line graph of a simple undirected weighted graph ...
- Problem 1566 - C - Spanning Tree 动态最小生成树
Problem 1566 - C - Spanning Tree 给出一个联通图,然后每次加一条边,每次需要求最小生成树 1 #include <iostream> 2 #include ...
随机推荐
- 关于用户与服务端密码的校验问题 !mysql php
问题:如何拿到服务端的数据与客户端的数据进行对比! 判断是否一致: 问题解决步骤: 建立数据库连接: $conn = mysqli_connect(主机地址,用户名,用户密码,数据库名字): 查询数据 ...
- 『Shell编程』学习记录(1)
例1. $ cat ex1 date pwd cd .. $ bash ex1 # 运行,显示当前日期和当前目录,但没有执行返回上级目录,因为执行的时候终端会产生一个子shell(类似于C语言调用函数 ...
- 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 二十║Vue基础终篇:传值+组件+项目说明
缘起 新的一天又开始啦,大家也应该看到我的标题了,是滴,Vue基础基本就到这里了,咱们回头看看这一路,如果你都看了,并且都会写了,那么现在你就可以自己写一个Demo了,如果再了解一点路由,ajax请求 ...
- Pycharm使用教程(四)-安装python依赖包(非常详细,非常实用)
简介 在做python开发时,需要很多依赖包,如果已经安装pip,安装依赖包,可以通过命令行:没有安装的,也可以通过PyCharm安装. 具体安装步骤 1.在File->Setting,如图: ...
- Python 闭包小记
闭包就是能够读取其他函数内部变量的函数.例如在javascript中,只有函数内部的子函数才能读取局部变量,所以闭包可以理解成“定义在一个函数内部的函数“.在本质上,闭包是将函数内部和函数外部连接起来 ...
- ionic4 混合移动开发 (前世今生)
ionic 从2016年初识,经历了 ionic2 ionic3.至今 ionic4,终于在2018年7月份发布了测试版. ionic Framework 可以说得上是最接近原生app的ui组件,漂亮 ...
- Exceptionless邮箱设置
在web.config中配置邮箱: <system.net> <mailSettings> <smtp from="xxx@163.com"> ...
- 【Android Studio安装部署系列】十四、Android studio移除工程和删除项目
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 概述 Android Studio删除工程.项目的操作步骤. 移除工程 主要用于从最近打开的项目列表中移除.硬盘中还是存在这个项目的. F ...
- eclipse升级Android SDK Tool版本到25.2.5后运行项目报错Unable to build: the file dx.jar was not loaded from the SDK folder
概述 由于最近通过SDK-Manager更新了build-tools,当要用到dx.jar这个包时,自动调用最新版本Android SDK build-tools中dx.jar,但是运行android ...
- GoLang structTag说明
在处理json格式字符串的时候,经常会看到声明struct结构的时候,属性的右侧还有小米点括起来的内容.形如 type User struct { UserId int `json:"use ...