\(\mathscr{Description}\)

  Link.

  在一个网格图内有 \(n\) 个格子有正价值,给出四种限制:横 / 纵坐标不大于 / 不小于 \(a\) 的格子不能选超过 \(b\) 个。求能选出格子价值之和的最大值。

  \(n\le80\),坐标范围 \(1\le X\le100\),限制数量 \(m\le320\)。

\(\mathscr{Solution}\)

  你看出来是网络流,但应当留意,图本身也许不是全部。

  大概感知一下,感觉是一个源点-横坐标-价值点-纵坐标-汇点的模型,但很难同时处理坐标上的前缀和限制与后缀和限制。怎么办?枚举最终选出的格子数 \(k\),它给我们带来了额外条件——对后缀的限制可以化归为对前缀的限制。

  具体地,以横坐标为例,先人为限制按横坐标升序选格子。对于 \(b<k\) 的某个限制,若限制前缀不超过 \(b\) 个,相当于选出的第 \(b+1..k\) 个格子的横坐标都得 \(>a\);若限制后缀不超过 \(b\) 个,相当于选出的第 \(1..k-b\) 个格子的横坐标都得 \(<a\)。纵坐标同理,规定纵坐标升序选即可,二者只是考虑角度不同,并不矛盾。依此跑费用流,若满流,用最大费用更新答案即可。

  复杂度 \(\mathcal O(n\operatorname{Dinic}(n,n^2))\),当然边数可以优化至 \(\mathcal O(n\log n)\)。

\(\mathscr{Code}\)

/*+Rainybunny+*/

#include <bits/stdc++.h>

#define rep(i, l, r) for (int i = l, rep##i = r; i <= rep##i; ++i)
#define per(i, r, l) for (int i = r, per##i = l; i >= per##i; --i) typedef long long LL;
typedef std::pair<int, LL> PIL;
#define fi first
#define se second template <typename Tp>
inline void chkmin(Tp& u, const Tp& v) { v < u && (u = v, 0); }
template <typename Tp>
inline void chkmax(Tp& u, const Tp& v) { u < v && (u = v, 0); }
template <typename Tp>
inline Tp imin(const Tp& u, const Tp& v) { return u < v ? u : v; }
template <typename Tp>
inline Tp imax(const Tp& u, const Tp& v) { return u < v ? v : u; } const int MAXN = 80, MAXM = 320, MAXX = 100, IINF = 0x3f3f3f3f;
const LL LINF = 1ll << 60; namespace FG { const int MAXND = MAXN * 2 + MAXX * 2 + 2;
const int MAXEG = MAXN * MAXN * 2 + MAXN * 3;
int S, T, ecnt = 1, head[MAXND + 5], curh[MAXND + 5];
LL dis[MAXND + 5];
struct Edge { int to, flw; LL cst; int nxt; } graph[MAXEG * 2 + 5]; inline void clear() {
ecnt = 1;
rep (i, S, T) head[i] = 0;
} inline void link(const int s, const int t, const int f, const LL c) {
// printf("%d %d %d,%lld\n", s, t, f, c);
graph[++ecnt] = { t, f, c, head[s] }, head[s] = ecnt;
graph[++ecnt] = { s, 0, -c, head[t] }, head[t] = ecnt;
} inline bool spfa() {
static std::queue<int> que;
static bool inq[MAXND + 5];
rep (i, S, T) dis[i] = LINF;
dis[S] = 0, que.push(S);
while (!que.empty()) {
int u = que.front(); que.pop(), inq[u] = false;
for (int i = head[u], v; i; i = graph[i].nxt) {
if (graph[i].flw && dis[v = graph[i].to] > dis[u] + graph[i].cst) {
dis[v] = dis[u] + graph[i].cst;
if (!inq[v]) que.push(v), inq[v] = true;
}
}
}
return dis[T] != LINF;
} inline PIL augment(const int u, int iflw) {
if (u == T) return { iflw, 0 };
static bool instk[MAXND + 5]; instk[u] = true;
PIL ret(0, 0);
for (int &i = curh[u], v; i; i = graph[i].nxt) {
if (graph[i].flw && !instk[v = graph[i].to]
&& dis[v] == dis[u] + graph[i].cst) {
PIL t(augment(v, std::min(iflw, graph[i].flw)));
ret.fi += t.fi, ret.se += t.se + t.fi * graph[i].cst;
graph[i].flw -= t.fi, graph[i ^ 1].flw += t.fi, iflw -= t.fi;
if (!iflw) break;
}
}
if (!ret.fi) dis[u] = LINF;
return instk[u] = false, ret;
} inline PIL dinic() {
PIL ret(0, 0);
while (spfa()) {
rep (i, S, T) curh[i] = head[i];
PIL t(augment(S, IINF));
ret.fi += t.fi, ret.se += t.se;
}
return ret;
} } // namespace FG. int n, m, jx[MAXN + 5], jy[MAXN + 5];
int lefx[MAXN + 5], rigx[MAXN + 5], lefy[MAXN + 5], rigy[MAXN + 5];
LL jv[MAXN + 5];
struct Restrict { int op, a, b; } lim[MAXM + 5]; int main() {
scanf("%d", &n);
rep (i, 1, n) scanf("%d %d %lld", &jx[i], &jy[i], &jv[i]);
scanf("%d", &m);
rep (i, 1, m) {
char op[5];
scanf("%s %d %d", op, &lim[i].a, &lim[i].b), lim[i].op = op[0];
} LL ans = 0;
rep (x, 1, n) {
// x = 4; // debug.
FG::clear();
FG::S = 0, FG::T = 2 * x + 2 * n + 1; rep (i, 1, x) {
FG::link(FG::S, i, 1, 0);
FG::link(i + x, FG::T, 1, 0);
}
rep (i, 1, n) FG::link(i + 2 * x, i + n + 2 * x, 1, -jv[i]); rep (i, 1, x) lefx[i] = lefy[i] = 1, rigx[i] = rigy[i] = MAXX;
rep (i, 1, m) if (lim[i].b < x) {
if (lim[i].op == 'L') chkmax(lefx[lim[i].b + 1], lim[i].a + 1);
if (lim[i].op == 'R') chkmin(rigx[x - lim[i].b], lim[i].a - 1);
if (lim[i].op == 'U') chkmin(rigy[x - lim[i].b], lim[i].a - 1);
if (lim[i].op == 'D') chkmax(lefy[lim[i].b + 1], lim[i].a + 1);
}
rep (i, 2, x) {
chkmax(lefx[i], lefx[i - 1]), chkmax(lefy[i], lefy[i - 1]);
}
per (i, x - 1, 1) {
chkmin(rigx[i], rigx[i + 1]), chkmin(rigy[i], rigy[i + 1]);
} rep (i, 1, x) {
rep (j, 1, n) {
if (lefx[i] <= jx[j] && jx[j] <= rigx[i]) {
FG::link(i, j + 2 * x, 1, 0);
}
if (lefy[i] <= jy[j] && jy[j] <= rigy[i]) {
FG::link(j + n + 2 * x, i + x, 1, 0);
}
}
} PIL res(FG::dinic());
if (res.fi == x) chkmax(ans, -res.se);
// break; // debug
}
printf("%lld\n", ans);
return 0;
}

Solution -「AGC 031E」Snuke the Phantom Thief的更多相关文章

  1. Solution -「AGC 036D」「AT 5147」Negative Cycle

    \(\mathcal{Descriprtion}\)   Link.   在一个含 \(n\) 个结点的有向图中,存在边 \(\lang i,i+1,0\rang\),它们不能被删除:还有边 \(\l ...

  2. Solution -「AGC 016F」Games on DAG

    \(\mathcal{Description}\)   Link.   给定一个含 \(n\) 个点 \(m\) 条边的 DAG,有两枚初始在 1 号点和 2 号点的棋子.两人博弈,轮流移动其中一枚棋 ...

  3. Solution -「AGC 026D」Histogram Coloring

    \(\mathcal{Description}\)   Link.   有 \(n\) 列下底对齐的方格纸排成一行,第 \(i\) 列有 \(h_i\) 个方格.将每个方格染成黑色或白色,求使得任意完 ...

  4. Solution -「AGC 004E」「AT 2045」Salvage Robots

    \(\mathcal{Description}\)   Link.   有一个 \(n\times m\) 的网格.每个格子要么是空的,要么有一个机器人,要么是一个出口(仅有一个).每次可以命令所有机 ...

  5. Solution -「AGC 012F」「AT 2366」Prefix Median

    \(\mathcal{Description}\)   Link.   给定序列 \(\{a_{2n-1}\}\),将 \(\{a_{2n-1}\}\) 按任意顺序排列后,令序列 \(b_i\) 为前 ...

  6. Solution -「AGC 010C」「AT 2304」Cleaning

    \(\mathcal{Description}\)   Link.   给定一棵 \(n\) 个点的无根树,点有点权,每次选择两个不同的叶子,使它们间的简单路径的所有点权 \(-1\),问能否将所有点 ...

  7. Solution -「AGC 019E」「AT 2704」Shuffle and Swap

    \(\mathcal{Description}\)   Link.   给定 \(01\) 序列 \(\{A_n\}\) 和 \(\{B_n\}\),其中 \(1\) 的个数均为 \(k\).记 \( ...

  8. Solution -「AGC 019F」「AT 2705」Yes or No

    \(\mathcal{Description}\)   Link.   有 \(n+m\) 个问题,其中 \(n\) 个答案为 yes,\(m\) 个答案为 no.每次你需要回答一个问题,然后得知这个 ...

  9. Solution -「AGC 013E」「AT 2371」Placing Squares

    \(\mathcal{Description}\)   Link.   给定一个长度为 \(n\) 的木板,木板上有 \(m\) 个标记点,第 \(i\) 个标记点距离木板左端点的距离为 \(x_i\ ...

  10. Solution -「AGC 003D」「AT 2004」Anticube

    \(\mathcal{Description}\)   Link.   给定 \(n\) 个数 \(a_i\),要求从中选出最多的数,满足任意两个数之积都不是完全立方数.   \(n\le10^5\) ...

随机推荐

  1. linux终端高级玩法详细介绍

    专注于收集整理更多好玩技巧 更改终端命令行颜色 vi /etc/profile PS1='[\[\e[32m\]\u\[\e[0m\]\[\e[35m\]@\[\e[0m\]\[\e[33m\]\h\ ...

  2. Windows安装Mysql后一段时间后Mysql服务无法启动的问题

    本人在windows重装电脑后遇到一个比较麻烦的问题一直没有解决,今日有幸看到某大佬的博客得以解决.真实万分感激,特来分享一下. 第一次安装Mysql8.0之后,此次安装是将整个mysql包进行安装, ...

  3. js 数字计算的精度问题

    〇.js 的数值计算存在结果不精确的情况 最近接触财务相关系统,页面上会有一些简单的计算,就发现其实是非常简单的计算,但 js 计算出来的结果却不是预期值,可能带上一大串 0 或 9,导致计算结果错误 ...

  4. 题解:CF687C The Values You Can Make

    CF687C The Values You Can Make 题解 题目翻译感觉不明不白的(至少我看了几遍没看懂),这里给个较为清晰的题面. 题目描述 给你 \(n\) 个硬币,第 \(i\) 个硬币 ...

  5. 【VMware VCF】通过备份的配置文件还原 SDDC Manager 组件。

    之前在这篇文章(使用 SFTP 服务器备份 VCF 核心组件的配置文件.)中配置并备份了 VCF 环境中 SDDC Manager 组件的配置文件,这篇文章接着这个主题,看看当 SDDC Manage ...

  6. 批量归一化(BN, Batch Normalization)

    现在的神经网络通常都特别深,在输出层向输入层传播导数的过程中,梯度很容易被激活函数或是权重以指数级的规模缩小或放大,从而产生"梯度消失"或"梯度爆炸"的现象,造 ...

  7. 2025年前端面试准备html篇

    时光飞逝,一晃已经工作了10年了,2014年一个人背着书包拉着箱子,下火车去做637路公交车的场景历历在目,637路公交车从起点坐到终点,开启了工作的第一站,这趟已经在路上行驶了10年的列车,经历多了 ...

  8. Redis【1】- 如何阅读 Redis源码

    1 Redis 的简介 Redis 实际上是简称,全称为 Remote Dictionary Server (远程字典服务器),由 Salvatore Sanfilippo 写的高性能 key-val ...

  9. SPRING 动态注册BEAN

    场景 有些情况下,不能直接使用BEAN的方式: @Bean(name = "storage") public DataSourceProxy storageDataSourcePr ...

  10. Java线程命名问题解决

    前言 网上冲浪时刷到线程池的文章,想想看自己好像还没在实际场景中设置过线程名称,小小研究一下. 研究过程 默认命名 创建的线程都会有自己的名字,如果不设置,程序会给线程默认的名字,如Thread-0 ...