【2018百度之星程序设计大赛初赛】degree

Problem Description

度度熊最近似乎在研究图论。给定一个有 N 个点 (vertex) 以及 M 条边 (edge) 的无向简单图 (undirected simple graph),此图中保证没有任何圈 (cycle) 存在。

现在你可以对此图依序进行以下的操作:

移除至多 K 条边。
在保持此图是没有圈的无向简单图的条件下,自由的添加边至此图中。
请问最后此图中度数 (degree) 最大的点的度数可以多大呢?

Input

输入的第一行有一个正整数 T,代表接下来有几笔测试资料。

对于每笔测试资料: 第一行有三个整数 N, M, K。 接下来的 M 行每行有两个整数 a 及 b,代表点 a 及 b 之间有一条边。 点的编号由 0 开始至 N-1。
0≤K≤M≤2×1050≤K≤M≤2×105​​
1≤N≤2×1051≤N≤2×105​​
0≤a,b<N0≤a,b<N
给定的图保证是没有圈的简单图1≤T≤231≤T≤23
至多 2 笔测试资料中的 N>1000N>1000

Output

对于每一笔测试资料,请依序各自在一行内输出一个整数,代表按照规定操作后可能出现的最大度数。

Sample Input

2
3 1 1
1 2
8 6 0
1 2
3 1
5 6
4 1
6 4
7 0

Sample Output

2
4

Solution

其实题目大意就是给你一个森林(多棵不相连的树),可以从中删去至多k条边,然后添加任意条边,但保证还是一个森林或一颗树,使得一个点的度数最大,让你求这个最大的度数。
先考虑不删边的情况,我们可以发现如果不删边,那么它的答案就是这n个点中度数最大的点的度数加上森林的个数再减一,减一是减去当前的这颗树。就相当于从每颗树都向这个最大度数的点所在的树连了一条边。之后我们考虑删边的情况,如果要删边,那么也就是相当于将除了这个最大度数的点向其他点连的边,其他边都要尽量删去,然后再尽可能多的向这个最大度数的点连一条边,这样就能保证在不损失最大度数的情况下再让答案更优。(将有最大度树的数向其他的树弄边,因为其他树是连起来的,所以会形成环,所以这个时候可以认为是删掉连向的那棵树内的一条边,因为最多可以删k条边,那么只需要从(m-度数最大的点的度数)和k中取最小值即可)

m-度数最大的点的度数 表示除了其他未度数最大的树相连的边

 

 

综上,答案就是度数最大的点的度数+森林的个数-1+min(m-度数最大的点的度数,k)。

PS 哇,这个题是水题,死了好几次,,,,,

#include<iostream>
#include<algorithm>
#include<cstring>
#include<vector>
#include<set>
using namespace std;
const int maxn = 2e5 + 10;
vector<int>e[maxn];
int n, m, k, T;
int x[maxn], cnt, vis[maxn];
void dfs(int s)
{
    for (int i = 0; i<e[s].size(); i++)
    {
        if (vis[e[s][i]]!=-1)
        {
            vis[e[s][i]] = cnt;
            dfs(e[s][i]);
        }
    }
}
int main()
{
    
    scanf("%d",&t);
    while (T--)
    {
        memset(vis,-1,sizeof(vis));
        for (int i = 0; i <= n; i++)
            e[i].clear();
        memset(vis, 0, sizeof(vis));
        memset(x, 0, sizeof(x));
        scanf("%d%d%d",&n,&m,&k);
        int a, b;
        for (int i = 0; i<m; i++)
        {
            scanf("%d%d",&a,&b);
            x[a]++;
            x[b]++;
            e[a].push_back(b);
            e[b].push_back(a);
        }
        cnt = 0;
        int ma = 0;
        for (int i = 0; i < n; i++)
        {
            if (ma<x[i])ma = x[i];
            if (vis[i]!=-1)
            {
                dfs(i);
                cnt++;
            }
        }
        int count = m - ma;//代表最多可以删除多少条边
        cnt -= 1;//除了那一块度最多的之外的联通快数量
        int ans = cnt + min(count, k) + ma;//答案等于孤立点的数量+当前的度数+最多可以去掉几条边(即可以再加入几条边)
        printf("%d\n",cnt);
    }
    return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值