F2 - Spanning Tree with One Fixed Degree - 并查集+DFS

该博客讨论了一道有趣的图论问题,要求在保持生成树性质的同时,确保1号节点的度数为K。解决方案涉及使用并查集来处理联通分量,并通过DFS或优化的并查集策略选择必要的边,以确保1号节点的度数满足条件。
摘要由CSDN通过智能技术生成

   这道题还是非常有意思的,题意很简单,就是给定一个图,和图上的双向边,要求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+7;
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=0; i<G[x].size(); i++)
    {
        nex=G[x][i];
        if (nex==1)continue;
        if (v[nex]==0)
        {
            v[nex]=1;
            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,0,sizeof(vis));
        memset(v,
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值