一个序列 \(d\{n\} = \{1\}\),有 \(m\) 种操作,每种操作都有一个操作次数的最大限制,且可以分为 \(4\) 类:
1. 将任意一个满足 \(d_i = a\) 的 \(d_i\) 改为 \(b\);
2. 将任意一个满足 \(d_i \in [a1, a2]\) 的 \(d_i\) 改为 \(b\);
3. 将任意一个满足 \(d_i = a\) 的 \(d_i\) 改为 \([b1, b2]\) 中的一个数;
4. 将任意一个满足 \(d_i \in [a1, a2]\) 的 \(d_i\) 改为 \([b1, b2]\) 中的一个数。
求最后序列中最多有多少个值为 \(k\) 的数。
\(n \le 10^7, k \le 2 \times 10^4, m \le 10^5\)。
不难看出这是一道网络流的题目,我们先考虑暴力。记源汇点分别为 \(s, t\),对于题目的每一种限制,我们直接暴力连边,容量为对应的限制,再连边 \((s, 1, n)\) 与 \((k, t, n)\),求从 \(s\) 到 \(t\) 的最大流即可。
真的是这样吗?其实不是,观察可以发现,题目要求的是总的次数不超过限制,而如果直接连边则是对于每一对 \(a, b\) 均不超过限制,而这是不对的,还有可能有多的但是题目数据太水了,这一点好像没有怎么体现。因此,对于每一个操作都应该新建两个虚拟节点 \(l\) 与 \(r\),对于所有的 \(a\),连边 \((a, l, \inf)\),对于所有的 \(b\),连边 \((r, b, \inf)\),再连边 \((l, r, lim)\) 即可,其中 \(lim\) 为这种操作的限制次数。这样总体的点数为 \(\mathcal{O}(k)\) 的,而总边数达到了惊人的 \(\mathcal{O}(mk ^ 2)\),这显然是不行的。
观察到题目给出的形式是点向点连边、区间向区间连边、点与区间互相连边,非常符合线段树优化建图的形式,于是我们直接套一个线段树优化建图的板子,就可以将点数与边数均优化至 \(\mathcal{O}(k \log k)\) 级别。
具体在实现时,我们队线段树上的每一个区间都赋予一个标号,然后在连边时,通过在线段树上 DFS,找出所有有用的区间标号并将其存储至一个栈里,然后再连边即可。
#include <bits/stdc++.h>
using i64 = long long;
constexpr int N = 1e5 + 50, inf = 0x3f3f3f3f, M = 1e6 + 50;
int n, m, k, cnt;
template<int N = 1000050, int M = 3000050>
struct Dinic {
static const int inf = 0x3f3f3f3f;
struct Edge {
int u, v, cap, flow;
Edge() = default;
Edge(int _u, int _v, int _c) { u = _u, v = _v, cap = _c, flow = 0; }
} edges[M];
int cnt = 0;
std::vector<int> adj[N];
int s, t, dis[N], cur[N];
void add(int u, int v, int cap) {
edges[cnt++] = Edge(u, v, cap), edges[cnt++] = Edge(v, u, 0);
adj[u].push_back(cnt - 2), adj[v].push_back(cnt - 1);
}
bool bfs() {
std::memset(cur, 0, sizeof(cur));
std::memset(dis, 0x3f, sizeof(dis));
std::queue<int> q;
q.push(s), dis[s] = 0;
while (!q.empty()) {
int u = q.front();
q.pop();
for (auto i : adj[u]) {
auto [_, v, cap, flow] = edges[i];
if (dis[v] == inf && cap > flow) dis[v] = dis[u] + 1, q.push(v);
}
}
return dis[t] != inf;
}
int dfs(int u, int lim) {
if (u == t || !lim) return lim;
int res = 0;
for (int &i = cur[u], f; i < adj[u].size(); i++) {
auto &[_, v, cap, flow] = edges[adj[u][i]];
if (dis[v] == dis[u] + 1 && (f = dfs(v, std::min(lim, cap - flow)))) {
res += f, lim -= f;
flow += f, edges[adj[u][i] ^ 1].flow -= f;
if (!lim) break;
}
}
return res;
}
int maxFlow(int _s, int _t) {
int res = 0;
s = _s, t = _t;
while (bfs()) res += dfs(s, inf);
return res;
}
};
Dinic solver;
int idx[M], rev[M];
void build(int s, int t, int u) {
idx[u] = ++cnt, rev[u] = ++cnt;
if (s == t) {
solver.add(idx[u], rev[u], inf), solver.add(rev[u], idx[u], inf);
return;
}
int mid = (s + t) >> 1, lc = u << 1, rc = u << 1 | 1;
build(s, mid, lc), build(mid + 1, t, rc);
solver.add(idx[u], idx[lc], inf), solver.add(idx[u], idx[rc], inf);
solver.add(rev[lc], rev[u], inf), solver.add(rev[rc], rev[u], inf);
}
int st[N], top;
void find(int l, int r, int s, int t, int u) {
if (l <= s && t <= r) {
st[++top] = u;
return;
}
int mid = (s + t) >> 1, lc = u << 1, rc = u << 1 | 1;
if (mid >= l) find(l, r, s, mid, lc);
if (mid < r) find(l, r, mid + 1, t, rc);
}
signed main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
std::cin >> n >> m >> k;
build(1, k, 1);
for (int i = 1; i <= m; i++) {
int opt, lim;
std::cin >> opt >> lim;
int ts = ++cnt, tt = ++cnt;
solver.add(ts, tt, lim);
if (opt == 1) {
int a, b;
std::cin >> a >> b;
find(a, a, 1, k, 1);
while (top) solver.add(rev[st[top--]], ts, inf);
find(b, b, 1, k, 1);
while (top) solver.add(tt, idx[st[top--]], inf);
} else if (opt == 2) {
int a1, a2, b;
std::cin >> a1 >> a2 >> b;
find(a1, a2, 1, k, 1);
while (top) solver.add(rev[st[top--]], ts, inf);
find(b, b, 1, k, 1);
while (top) solver.add(tt, idx[st[top--]], inf);
} else if (opt == 3) {
int a, b1, b2;
std::cin >> a >> b1 >> b2;
find(a, a, 1, k, 1);
while (top) solver.add(rev[st[top--]], ts, inf);
find(b1, b2, 1, k, 1);
while (top) solver.add(tt, idx[st[top--]], inf);
} else {
int a1, a2, b1, b2;
std::cin >> a1 >> a2 >> b1 >> b2;
find(a1, a2, 1, k, 1);
while (top) solver.add(rev[st[top--]], ts, inf);
find(b1, b2, 1, k, 1);
while (top) solver.add(tt, idx[st[top--]], inf);
}
}
int s = ++cnt, t = ++cnt;
find(1, 1, 1, k, 1);
int p1 = st[top];
top = 0;
find(k, k, 1, k, 1);
int pk = st[top];
top = 0;
solver.add(s, rev[p1], n), solver.add(idx[pk], t, n);
std::cout << solver.maxFlow(s, t) << "\n";
return 0;
}
标签:std,solver,int,ill,P5029,add,Over,inf,top
From: https://www.cnblogs.com/forgot-dream/p/17608132.html