回归第八题



无向图缩点,不知道为啥我写tarjan就是过不了
注意最后一定要把缩点后的大小按照从大到小开始删边
比如说你删4条,在一个环中可以另外得到3个分量,但是如果放在两个环里面分别为删两边,则总和只能得到2个分量
我的代码只能过80的点(我真的尽力了,现在晚上1.55我困得哟死)

#include
using namespace std;
#define lowbit(x) x&(-x)
#define ll long long
const int maxn=1e6+5;
vectorQ[maxn];
int n,m,k,st,cnt,sz;
int a[maxn],b[maxn],S[maxn],dfn[maxn],low[maxn],stk[maxn],sum[maxn];
void dfs(int now,int fa){
	dfn[now]=low[now]=++cnt;
	stk[++st]=now;
	for(int i=0;i#include 
#include 
#define MAXN 4000010
struct edge{
    int to, next;
} e[MAXN];
int head[MAXN], tot = 0;
void add_edge(int u, int v) {
    tot++; e[tot].to = v, e[tot].next = head[u]; head[u] = tot;
}
int dis[MAXN], circle[MAXN], cnt = 0, size = 0;
void dfs(int node, int fa) {
    dis[node] = dis[fa] + 1;
    for (int i = head[node]; i; i = e[i].next) {
        int y = e[i].to;
        if (y == fa) continue;
        if (!dis[y]) {
            dfs(y, node);
        } else if (dis[y] < dis[node] + 1) {
            cnt++; circle[cnt] = dis[node] - dis[y] + 1;
            size += circle[cnt];
        }
    }
}
int main() {
    int n, m, k; scanf("%d %d %d", &n, &m, &k);
    for (int i = 1; i <= m; i++) {
        int u, v; scanf("%d %d", &u, &v);
        add_edge(u, v); add_edge(v, u);
    }
    int kuai = 0;
    for (int i = 1; i <= n; i++) {
        if (!dis[i]) {
            dis[i] = 1, kuai++;
            dfs(i, 0);
        }
    }
    if (m - size >= k) {
        printf("%d\n", kuai + k);
        return 0;
    }
    std::sort(circle + 1, circle + 1 + cnt);
    int ans = m - size + kuai; k -= m - size;
    for (int i = cnt; i >= 1; i--) {
        if (k - circle[i] >= 0) {
            ans += circle[i] - 1;
        } else {
            ans += k - 1;
            break;
        }
        k -= circle[i];
        if (k <= 0) break;
    }
    printf("%d\n", ans);
    return 0;
}