結果
| 問題 |
No.1479 Matrix Eraser
|
| コンテスト | |
| ユーザー |
|
| 提出日時 | 2025-05-10 15:55:17 |
| 言語 | C++23 (gcc 13.3.0 + boost 1.87.0) |
| 結果 |
AC
|
| 実行時間 | 877 ms / 3,000 ms |
| コード長 | 9,561 bytes |
| コンパイル時間 | 4,453 ms |
| コンパイル使用メモリ | 320,672 KB |
| 実行使用メモリ | 91,976 KB |
| 最終ジャッジ日時 | 2025-05-10 15:55:37 |
| 合計ジャッジ時間 | 19,443 ms |
|
ジャッジサーバーID (参考情報) |
judge2 / judge3 |
(要ログイン)
| ファイルパターン | 結果 |
|---|---|
| sample | AC * 2 |
| other | AC * 39 |
ソースコード
// competitive-verifier: PROBLEM https://yukicoder.me/problems/no/1479
#include <algorithm>
#include <iterator>
#include <vector>
/// @brief 座標圧縮
template <class T>
struct coordinate_compression {
coordinate_compression() = default;
coordinate_compression(const std::vector<T> &_data) : data(_data) { build(); }
const T &operator[](int i) const { return data[i]; }
T front() const { return data.front(); }
T back() const { return data.back(); }
void add(T x) { data.emplace_back(x); }
void build() {
std::sort(data.begin(), data.end());
data.erase(std::unique(data.begin(), data.end()), data.end());
}
bool exists(T x) const {
auto it = std::lower_bound(data.begin(), data.end(), x);
return it != data.end() && *it == x;
}
int get(T x) const {
return std::distance(data.begin(), std::lower_bound(data.begin(), data.end(), x));
}
int lower_bound(T x) const {
return std::distance(data.begin(), std::lower_bound(data.begin(), data.end(), x));
}
int upper_bound(T x) const {
return std::distance(data.begin(), std::upper_bound(data.begin(), data.end(), x));
}
std::vector<int> compress(const std::vector<T> &v) const {
int n = v.size();
std::vector<int> res(n);
for (int i = 0; i < n; ++i) res[i] = get(v[i]);
return res;
}
int size() const { return data.size(); }
private:
std::vector<T> data;
};
/// @brief 座標圧縮
template <class T>
std::vector<int> compress(const std::vector<T> &v) {
coordinate_compression cps(v);
std::vector<int> res;
res.reserve(std::size(v));
for (auto &&x : v) res.emplace_back(cps.get(x));
return res;
}
#include <cassert>
#include <limits>
#include <queue>
/**
* @brief 最大流
*
* @tparam Cap
*/
template <class Cap>
struct mf_graph {
mf_graph() : _n(0) {}
explicit mf_graph(int n) : _n(n), g(n) {}
int size() const { return _n; }
int add_edge(int from, int to, Cap cap) {
assert(0 <= from && from < _n);
assert(0 <= to && to < _n);
assert(0 <= cap);
int m = int(pos.size());
pos.emplace_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].emplace_back(to, to_id, cap);
g[to].emplace_back(from, from_id, 0);
return m;
}
struct edge {
int from, to;
Cap cap, flow;
};
edge get_edge(int i) {
int m = int(pos.size());
assert(0 <= i && i < m);
auto _e = g[pos[i].first][pos[i].second];
auto _re = g[_e.to][_e.rev];
return edge{pos[i].first, _e.to, _e.cap + _re.cap, _re.cap};
}
std::vector<edge> edges() {
int m = int(pos.size());
std::vector<edge> result;
for (int i = 0; i < m; ++i) result.emplace_back(get_edge(i));
return result;
}
void change_edge(int i, Cap new_cap, Cap new_flow) {
int m = int(pos.size());
assert(0 <= i && i < m);
assert(0 <= new_flow && new_flow <= new_cap);
auto &_e = g[pos[i].first][pos[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) {
assert(0 <= s && s < _n);
assert(0 <= t && t < _n);
assert(s != t);
std::vector<int> level(_n), iter(_n);
auto bfs = [&]() {
std::fill(level.begin(), level.end(), -1);
level[s] = 0;
std::queue<int> que;
que.emplace(s);
while (!que.empty()) {
int v = que.front();
que.pop();
for (auto e : g[v]) {
if (e.cap == 0 || level[e.to] >= 0) continue;
level[e.to] = level[v] + 1;
if (e.to == t) return;
que.emplace(e.to);
}
}
};
auto dfs = [&](auto self, int v, Cap up) {
if (v == s) return up;
Cap res = 0;
int level_v = level[v];
for (int &i = iter[v]; i < int(g[v].size()); ++i) {
_edge &e = g[v][i];
if (level_v <= level[e.to] || g[e.to][e.rev].cap == 0) continue;
Cap d = self(self, e.to, std::min(up - 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 == up) 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;
}
std::vector<bool> min_cut(int s) {
std::vector<bool> visited(_n);
std::queue<int> que;
que.emplace(s);
while (!que.empty()) {
int p = que.front();
que.pop();
visited[p] = true;
for (auto e : g[p]) {
if (e.cap && !visited[e.to]) {
visited[e.to] = true;
que.emplace(e.to);
}
}
}
return visited;
}
private:
int _n;
struct _edge {
int to, rev;
Cap cap;
constexpr _edge(int _to, int _rev, Cap _cap) : to(_to), rev(_rev), cap(_cap) {}
};
std::vector<std::pair<int, int>> pos;
std::vector<std::vector<_edge>> g;
};
#ifdef ATCODER
#pragma GCC target("sse4.2,avx512f,avx512dq,avx512ifma,avx512cd,avx512bw,avx512vl,bmi2")
#endif
#pragma GCC optimize("Ofast,fast-math,unroll-all-loops")
#include <bits/stdc++.h>
#ifndef ATCODER
#pragma GCC target("sse4.2,avx2,bmi2")
#endif
template <class T, class U>
constexpr bool chmax(T &a, const U &b) {
return a < (T)b ? a = (T)b, true : false;
}
template <class T, class U>
constexpr bool chmin(T &a, const U &b) {
return (T)b < a ? a = (T)b, true : false;
}
constexpr std::int64_t INF = 1000000000000000003;
constexpr int Inf = 1000000003;
constexpr double EPS = 1e-7;
constexpr double PI = 3.14159265358979323846;
#define FOR(i, m, n) for (int i = (m); i < int(n); ++i)
#define FORR(i, m, n) for (int i = (m)-1; i >= int(n); --i)
#define FORL(i, m, n) for (int64_t i = (m); i < int64_t(n); ++i)
#define rep(i, n) FOR (i, 0, n)
#define repn(i, n) FOR (i, 1, n + 1)
#define repr(i, n) FORR (i, n, 0)
#define repnr(i, n) FORR (i, n + 1, 1)
#define all(s) (s).begin(), (s).end()
struct Sonic {
Sonic() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
std::cout << std::fixed << std::setprecision(20);
}
constexpr void operator()() const {}
} sonic;
using namespace std;
using ll = std::int64_t;
using ld = long double;
template <class T, class U>
std::istream &operator>>(std::istream &is, std::pair<T, U> &p) {
return is >> p.first >> p.second;
}
template <class T>
std::istream &operator>>(std::istream &is, std::vector<T> &v) {
for (T &i : v) is >> i;
return is;
}
template <class T, class U>
std::ostream &operator<<(std::ostream &os, const std::pair<T, U> &p) {
return os << '(' << p.first << ',' << p.second << ')';
}
template <class T>
std::ostream &operator<<(std::ostream &os, const std::vector<T> &v) {
for (auto it = v.begin(); it != v.end(); ++it) os << (it == v.begin() ? "" : " ") << *it;
return os;
}
template <class Head, class... Tail>
void co(Head &&head, Tail &&...tail) {
if constexpr (sizeof...(tail) == 0) std::cout << head << '\n';
else std::cout << head << ' ', co(std::forward<Tail>(tail)...);
}
template <class Head, class... Tail>
void ce(Head &&head, Tail &&...tail) {
if constexpr (sizeof...(tail) == 0) std::cerr << head << '\n';
else std::cerr << head << ' ', ce(std::forward<Tail>(tail)...);
}
void Yes(bool is_correct = true) { std::cout << (is_correct ? "Yes\n" : "No\n"); }
void No(bool is_not_correct = true) { Yes(!is_not_correct); }
void YES(bool is_correct = true) { std::cout << (is_correct ? "YES\n" : "NO\n"); }
void NO(bool is_not_correct = true) { YES(!is_not_correct); }
void Takahashi(bool is_correct = true) { std::cout << (is_correct ? "Takahashi" : "Aoki") << '\n'; }
void Aoki(bool is_not_correct = true) { Takahashi(!is_not_correct); }
int main(void) {
int n, m;
cin >> n >> m;
vector a(n, vector(m, 0));
cin >> a;
mf_graph<int> mf(n * m * 3 + n + m + 2);
int st = mf.size() - 2, gl = st + 1;
rep (i, n) {
rep (j, m) {
mf.add_edge(st, n * m * 3 + i, 1);
mf.add_edge(n * m * 3 + i, n * m + i * m + j, 1);
mf.add_edge(n * m * 2 + i * m + j, n * m * 3 + n + j, 1);
mf.add_edge(n * m * 3 + n + j, gl, 1);
}
}
rep (i, n) {
auto v = compress(a[i]);
rep (j, m) {
if (a[i][j])
mf.add_edge(n * m + i * m + v[j], i * m + j, 1);
}
}
rep (j, m) {
vector<int> u;
rep (i, n) u.emplace_back(a[i][j]);
auto v = compress(u);
rep (i, n) {
if (a[i][j])
mf.add_edge(i * m + j, n * m * 2 + v[i] * m + j, 1);
}
}
co(mf.flow(st, gl));
return 0;
}