Library

This documentation is automatically generated by online-judge-tools/verification-helper

View the Project on GitHub Kyo-s-s/Library

:heavy_check_mark: Test/yosupo-Matching-on-Bipartite-Graph.test.cpp

Depends on

Code

#include <bits/stdc++.h>
using namespace std;

#define PROBLEM "https://judge.yosupo.jp/problem/bipartitematching"

#include "../Data_structure/Dinic.hpp"

int main() {
    
    int L, R, M;
    cin >> L >> R >> M;    

    Dinic<long long> dinic(L + R + 2);
    int s = L + R, t = L + R + 1;

    vector<int> idxs(M), A(M), B(M);

    for (int i = 0; i < M; i++) {
        int a, b;
        cin >> a >> b;
        int idx = dinic.add_edge(a, L + b, 1);
        idxs[i] = idx;
        A[i] = a;
        B[i] = b;
    }

    for (int i = 0; i < L; i++) {
        dinic.add_edge(s, i, 1);
    }
    for (int i = 0; i < R; i++) {
        dinic.add_edge(L + i, t, 1);
    }

    int ans = dinic.flow(s, t);
    cout << ans << "\n";
    for (int i = 0; i < M; i++) {
        int e = dinic.get_flow(idxs[i]);
        if (e == 1) {
            cout << A[i] << " " << B[i] << "\n";
        }
    }
}
#line 1 "Test/yosupo-Matching-on-Bipartite-Graph.test.cpp"
#include <bits/stdc++.h>
using namespace std;

#define PROBLEM "https://judge.yosupo.jp/problem/bipartitematching"

#line 1 "Data_structure/Dinic.hpp"
template<class Cap> struct Dinic {
  public:
    Dinic(int n) : n(n), g(n) {}

    int add_edge(int from, int to, Cap cap) {
        int m = (int)edges.size();
        edges.push_back({from, (int)g[from].size()});
        int from_id = (int)g[from].size();
        int to_id = (int)g[to].size();
        if (from == to) to_id++;
        g[from].push_back(dinic_edge{to, to_id, cap});
        g[to].push_back(dinic_edge{from, from_id, 0});
        return m;
    }

    void change_edge(int i, Cap new_cap, Cap new_flow) {
        auto& e = g[edges[i].first][edges[i].second];
        auto& re = g[e.to][e.rev];
        e.cap = new_cap - new_flow;
        re.cap = new_flow;
    }

    Cap flow(int s, int t) {
        return flow(s, t, std::numeric_limits<Cap>::max());
    }

    Cap flow(int s, int t, Cap flow_limit) {
        std::vector<int> level(n), iter(n);
        simple_queue<int> que;

        auto bfs = [&]() {
            std::fill(level.begin(), level.end(), -1);
            level[s] = 0;
            que.clear();
            que.push(s);
            while (!que.empty()) {
                int v = que.pop_front();
                for (auto e : g[v]) {
                    if (e.cap > 0 && level[e.to] == -1) {
                        level[e.to] = level[v] + 1;
                        if (e.to == t) return;
                        que.push(e.to);
                    }
                }
            }
        };

        auto dfs = [&](auto dfs, int v, Cap limit) -> Cap {
            if (v == s) return limit;
            Cap res = 0;
            int level_v = level[v];
            for (int& i = iter[v]; i < int(g[v].size()); i++) {
                dinic_edge& e = g[v][i];
                if (level_v <= level[e.to] || g[e.to][e.rev].cap == 0) continue;
                Cap d = dfs(dfs, e.to, std::min(limit - res, g[e.to][e.rev].cap));
                if (d <= 0) continue;
                g[v][i].cap += d;
                g[e.to][e.rev].cap -= d;
                res += d;
                if (res == limit) return res;
            }
            level[v] = n;
            return res;
        };

        Cap flow = 0;
        while (flow < flow_limit) {
            bfs();
            if (level[t] == -1) break;
            std::fill(iter.begin(), iter.end(), 0);
            Cap f = dfs(dfs, t, flow_limit - flow);
            if (!f) break;
            flow += f;
        }
        return flow;
    }

    Cap get_flow(int i) {
        auto _e = g[edges[i].first][edges[i].second];
        auto _re = g[_e.to][_e.rev];
        return _re.cap;
    } 

    std::vector<bool> min_cut(int s) {
        std::vector<bool> visited(n);
        simple_queue<int> que;
        que.push(s);
        while (!que.empty()) {
            int p = que.pop_front();
            visited[p] = true;
            for (auto e : g[p]) {
                if (e.cap && !visited[e.to]) {
                    visited[e.to] = true;
                    que.push(e.to);
                }
            }
        }
        return visited;
    }

  private: 
    int n;
    struct dinic_edge {
        int to, rev;
        Cap cap;
    };
    template<class T> struct simple_queue {
        std::vector<T> data;
        int pos = 0;
        void clear() {
            data.clear();
            pos = 0;
        }
        void push(T val) {
            data.emplace_back(val);
        }
        T pop_front() {
            return data[pos++];
        }
        bool empty() {
            return pos == (int)data.size();
        }
    };
    std::vector<std::pair<int, int>> edges;
    std::vector<std::vector<dinic_edge>> g;
};
#line 7 "Test/yosupo-Matching-on-Bipartite-Graph.test.cpp"

int main() {
    
    int L, R, M;
    cin >> L >> R >> M;    

    Dinic<long long> dinic(L + R + 2);
    int s = L + R, t = L + R + 1;

    vector<int> idxs(M), A(M), B(M);

    for (int i = 0; i < M; i++) {
        int a, b;
        cin >> a >> b;
        int idx = dinic.add_edge(a, L + b, 1);
        idxs[i] = idx;
        A[i] = a;
        B[i] = b;
    }

    for (int i = 0; i < L; i++) {
        dinic.add_edge(s, i, 1);
    }
    for (int i = 0; i < R; i++) {
        dinic.add_edge(L + i, t, 1);
    }

    int ans = dinic.flow(s, t);
    cout << ans << "\n";
    for (int i = 0; i < M; i++) {
        int e = dinic.get_flow(idxs[i]);
        if (e == 1) {
            cout << A[i] << " " << B[i] << "\n";
        }
    }
}
Back to top page