前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【小码匠自习室】并查集裸题:ABC288 - C - Don’t be cycle

【小码匠自习室】并查集裸题:ABC288 - C - Don’t be cycle

作者头像
小码匠
发布2023-03-06 14:30:34
3480
发布2023-03-06 14:30:34
举报

【小码匠自习室】并查集板子题:ABC288 - C - Don’t be cycle,亮点在参考题解

补题

周六的时候和小姨去吃饭了,晚上没能来的及线上赛,周日继续补题。

题目不难,难在我把并查集的板子打错了,稍微调试了几分钟。

本题的亮点:请移步参考题解, 后面我抽时间在研究

题目

  • https://atcoder.jp/contests/abc288/tasks/abc288_c

N顶点M边的简单无向图。顶点编号为1到N,第i边为顶点A

_i

和顶点B

_i

连接。从该图表中删除0条以上的几条边,使图表不具有闭路时,请求出删除边的根数的最小值。

制約
1 \leq N \leq 2 \times 10^5
0 \leq M \leq 2 \times 10^5
1 \leq A_i, B_i \leq N
  • 给定的图表很单纯图
  • 所有输入都是整数

入力
  • N M
  • A
_1

B

_1
  • A
_2

B

_2
\vdots
  • A
_M

B

_M
出力

输出答案。


入力例 1
代码语言:javascript
复制
6 7
1 2
1 3
2 3
4 2
6 5
4 6
4 5
出力例 1
代码语言:javascript
复制
2

可以通过删除连接顶点1和顶点2的两条边、顶点4和连接顶点5的两条边等方法使图表不具有闭路。

1条以下边的删除不能使图表没有闭路,所以输出2。


入力例 2
代码语言:javascript
复制
4 2
1 2
3 4
出力例 2
代码语言:javascript
复制
0

入力例 3
代码语言:javascript
复制
5 3
1 2
1 3
2 3
出力例 3
代码语言:javascript
复制
1

小码匠

思路
  • 采用并查集即可
代码
代码语言:javascript
复制
#include <bits/stdc++.h>
using namespace std;
#define endl '\n';
 
struct edge {
    int from;
    int to;
};
 
struct UF {
    vector<int> fa;
 
    UF (int n) :
                fa(n) {}
 
    void initialize (int n) {
        for (int i = 0; i < n; ++i) {
            fa[i] = i;
        }
    }
 
    int find(int a) {
        if (a != fa[a]) {
            fa[a] = find(fa[a]);
        }
        return fa[a];
    }
 
    void unite(int x, int y) {
        int a = find(x);
        int b = find(y);
        if (a != b) {
            fa[b] = a;
        }
    }
 
    bool is_same(int x, int y) {
        return find(x) == find(y);
    }
};
 
void best_coder() {
    int n, m;
    cin >> n >> m;
    queue<edge> g;
    UF uf(n);
    uf.initialize(n);
    for (int i = 0; i < m; ++i) {
        int a, b;
        cin >> a >> b;
        --a;
        --b;
        g.push({a, b});
    }
    int ans = 0;
    while (!g.empty()) {
        int a = g.front().from;
        int b = g.front().to;
        g.pop();
        if (uf.is_same(a, b)) {
            ++ans;
            continue;
        }
        uf.unite(a, b);
    }
    cout << ans;
}
 
void happy_coder() {
}
 
int main() {
    // 提升cin、cout效率
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cout.tie(nullptr);
 
    // 小码匠
    best_coder();
 
    // 最优解
    // happy_coder();
 
    // 返回
    return 0;
}
参考题解
代码语言:javascript
复制
#include <iostream>
#include <string>
#include <queue>
#include <stack>
#include <algorithm>
#include <list>
#include <vector>
#include <complex>
#include <utility>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <climits>
#include <bitset>
#include <ctime>
#include <map>
#include <unordered_map>
#include <set>
#include <unordered_set>
#include <cassert>
#include <cstddef>
#include <iomanip>
#include <numeric>
#include <tuple>
#include <sstream>
#include <fstream>
#include <functional>
 
using namespace std;
#define REP(i, n) for (int (i) = 0; (i) < (n); (i)++)
#define FOR(i, a, b) for (int (i) = (a); (i) < (b); (i)++)
#define RREP(i, a) for(int (i) = (a) - 1; (i) >= 0; (i)--)
#define FORR(i, a, b) for(int (i) = (a) - 1; (i) >= (b); (i)--)
#define DEBUG(C) cerr << #C << " = " << C << endl;
using LL = long long;
using VI = vector<int>;
using VVI = vector<VI>;
using VL = vector<LL>;
using VVL = vector<VL>;
using VD = vector<double>;
using VVD = vector<VD>;
using PII = pair<int, int>;
using PDD = pair<double, double>;
using PLL = pair<LL, LL>;
using VPII = vector<PII>;
template<typename T> using VT = vector<T>;
#define ALL(a) begin((a)), end((a))
#define RALL(a) rbegin((a)), rend((a))
#define SORT(a) sort(ALL((a)))
#define RSORT(a) sort(RALL((a)))
#define REVERSE(a) reverse(ALL((a)))
#define MP make_pair
#define FORE(a, b) for (auto &&a : (b))
#define EB emplace_back
#define GREATER(T) T, VT<T>, greater<T>
 
template<typename T>
inline bool chmax(T &a, T b) {
    if (a < b) {
        a = b;
        return true;
    }
    return false;
}
 
template<typename T>
inline bool chmin(T &a, T b) {
    if (a > b) {
        a = b;
        return true;
    }
    return false;
}
 
const int INF = 1e9;
const int MOD = INF + 7;
const LL LLINF = 1e18;
 
// UnionFind,素集合データ構造,DisjointSet
 
template<typename weight_type = int>
class union_find {
    static_assert(
            std::is_arithmetic<weight_type>::value,
            "Invalid weight type");
private:
    const std::function<void(int, int)> noop = [](int aa, int bb) {};
 
    std::vector<int> uni;
    std::vector<int> edge_count_;
    std::vector<weight_type> weights;
    int size_;
 
    const void check(const int n) const {
        if (this->group_size() < 0) {
            assert(false);
        }
        if (!(0 <= n && n < this->all_size())) {
            assert(false);
        }
    }
 
public:
    union_find() : uni(0), edge_count_(0), weights(0), size_(-1) {}
 
    union_find(const int n)
            : uni(n, -1), edge_count_(n), weights(n), size_(n) {
        this->check(n - 1);
    }
 
    union_find(const std::vector<weight_type> &_weights)
            : uni(_weights.size(), -1), edge_count_(_weights.size()),
              weights(_weights), size_(_weights.size()) {
        this->check((int) _weights.size() - 1);
    }
 
    bool same(const int a, const int b) {
        this->check(a);
        this->check(b);
        return this->find(a) == this->find(b);
    }
 
    int find(const int a) {
        this->check(a);
        return this->uni[a] < 0 ?
               a :
                this->uni[a] = this->find(this->uni[a]);
    }
 
    bool unite(int a, int b) {
        return unite(a, b, noop);
    }
 
    bool unite(int a, int b, const std::function<void(int, int)> &f) {
        a = this->find(a);
        b = this->find(b);
        this->edge_count_[a]++;
        if (a == b) {
            return false;
        }
        this->size_--;
        if (this->uni[a] > this->uni[b]) {
            std::swap(a, b);
        }
        f(a, b);
        this->uni[a] += this->uni[b];
        this->weights[a] += this->weights[b];
        this->edge_count_[a] += this->edge_count_[b];
        this->uni[b] = a;
        return true;
    }
 
    const int group_size() const {
        return this->size_;
    }
 
    const int all_size() const {
        return this->uni.size();
    }
 
    const int size(const int a) {
        this->check(a);
        return -this->uni[this->find(a)];
    }
 
    const int edge_count(const int a) {
        this->check(a);
        return this->edge_count_[this->find(a)];
    }
 
    const weight_type weight(const int a) {
        this->check(a);
        return this->weights[this->find(a)];
    }
};
 
 
void solve() {
    int N, M;
    cin >> N >> M;
    union_find<> uf(N);
    for (int i = 0; i < M; ++i) {
        int a, b;
        cin >> a >> b;
        uf.unite(a - 1, b - 1);
    }
 
    int ok_cnt = 0;
    set<int> checked;
    for (int i = 0; i < N; ++i) {
        const auto par = uf.find(i);
        if (checked.count(par)) continue;
        checked.emplace(par);
        ok_cnt += uf.size(par) - 1;
    }
 
    cout << max(M - ok_cnt, 0) << "\n";
}
 
int main(void) {
#ifndef ONLINE_JUDGE
    const auto in_stream = freopen("../in.txt", "r", stdin);
    if (in_stream == nullptr) {
        cerr << "ERROR!" << endl;
        return 1;
    }
#endif
 
    solve();
 
#ifndef ONLINE_JUDGE
    fclose(in_stream);
#endif
}
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2023-02-06,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 小码匠和老码农 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 【小码匠自习室】并查集板子题:ABC288 - C - Don’t be cycle,亮点在参考题解
    • 补题
      • 题目
        • 制約
        • 入力
        • 出力
        • 入力例 1
        • 出力例 1
        • 入力例 2
        • 出力例 2
        • 入力例 3
        • 出力例 3
      • 小码匠
        • 思路
        • 代码
        • 参考题解
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档