2018百度之星初赛(B)degree

degree

http://bestcoder.hdu.edu.cn/contests/contest_showproblem.php?cid=826&pid=1001

Accepts: 1581

Submissions: 3494

Time Limit: 2000/1000 MS (Java/Others)

Memory Limit: 131072/131072 K (Java/Others)

Problem Description

度度熊最近似乎在研究图论。给定一个有 N 个点 (vertex) 以及 M 条边 (edge) 的无向简单图 (undirected simple graph),此图中保证没有任何圈 (cycle) 存在。 现在你可以对此图依序进行以下的操作: 1. 移除至多 K 条边。 2. 在保持此图是没有圈的无向简单图的条件下,自由的添加边至此图中。 请问最后此图中度数 (degree) 最大的点的度数可以多大呢?

Input

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

Output

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

Sample Input

Copy

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

Sample Output

Copy

2
4

题解:

沒有圈 (cycle) 的简单图 (undirected simple graph),等价于由多颗树 (tree) 组成的森林 (forest)。这里用 V 代表点的数量,E 代表边的数量 (取代题目中的 N 以及 M),C 代表森林中树的数量。 

## K=0 的 case 不妨先化简一下题目,在 K=0 的状态下要达到 degree 最大化,可以贪心的把森林中所有树各自接一条边到已知 degree 最大的点上。答案是 C + 已知最大的 degree - 1。 

## K≥0 的 case 回到原题,題目中规定一定要在添加边之前把拔边的操作作完,但是实际上任意调换添加边以及拔掉边的顺序不会影响最后的结果。考虑贪心添加完边的树,可以多拔掉一条边再重新接上的效果等同于把答案 +1,要注意的是如果答案已经到达最大值 V-1 的话,那拔边再接上并不会影响答案。所以答案為min(V - 1, K + C + 已知最大的 degree - 1)。 

## 树的数量 C = V - E 由于给定的图是面数 (face) 为 1 的平面图 (planar graph),所以根据平面图的公式 V-E+F=C+1 整理后 C=V-E。另外一个证明:图中每个 connected component 都是由树所组成,也就是说每个 component 中边数会是点数 - 1,直接可推得 C=V?E。 有了 C=V-E 后,答案就变成 min(V - 1, K + V - E + 已知最大的 degree - 1)。也就是,我们其实不用真正的写出计算 connected component 的算法,只要统计 degree 最大的点有多大就可以计算出答案了。 整个时间复杂度为 O(V),注意这题中有 O(V)=O(E)。

代码:

#include<iostream>
#include<cstring>
#include<cmath>
#include<iomanip>
#include<algorithm>
#include<cstdio>
#define inf 0x3f3f3f3f
using namespace std;

int s[200005];

int main()
{
    int t;
    int n,m,k;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d%d",&n,&m,&k);
        memset(s,0,sizeof(s));
        int a,b;
        for(int i=0;i<m;i++)
        {
            cin>>a>>b;
            s[a]++;
            s[b]++;
        }
        int maxx=0;
        for(int i=0;i<n;i++)
        {
            if(maxx<s[i])
                maxx=s[i];
        }
        //cout<<"    "<<maxx<<endl;
        int c=n-m;  //树的数量
        if(k==0)
            cout<<c+maxx-1<<endl;
        else
            cout<<min(n-1,k+c+maxx-1)<<endl;
    }
    return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值