\(\mathscr{Description}\)

  Link.

  给定 \(n,k,m_s,m_e\) 和两个长为 \(n\) 的序列 \(\{s\},\{e\}\), 选择一个 \(S\subseteq[1,n]\cap\mathbb N\), 满足 \(\forall l\in[1,n-k+1],~\sum_{i=l}^{l+k-1}[i\in S]\in[m_s,k-m_e]\), 并最大化 \(\sum_{i=1}^n([i\in S]s_i+[i\notin S]e_i)\).

  \(n\le10^3\).

\(\mathscr{Solution}\)

  这里一个机械化方法的尝试, 没写代码是因为不可告人的原因. (

  我们想求

\[\begin{array}{rll}
\max & \displaystyle \sum_{i=1}^ne_i+\sum_{i=1}^n(s_i-e_i)(x_i-x_{i-1})\\
\text{s.t.} & x_0=0\\
& x_i\ge x_{i-1} & (\forall i>0)\\
& x_{i-1}-x_{i}\ge-1 & (\forall i>0) \\
& x_i-x_{i-k}\ge m_s & (\forall i\ge k)\\
& x_{i-k}-x_i\ge m_e-k & (\forall i\ge k)
\end{array}\tag1
\]

  先忽略 \(x_0=0\) 这个约束. 从现在起, 我们停止思考, 直接去套这个 trick. 通过足够大的代价 \(I\) 间接限定 \(\ge\):

\[\begin{aligned}
(1)\Leftrightarrow \max\quad & \sum_{i=1}^n e_i+\sum_{i=1}^n(s_i-s_{i+1}-e_i+e_{i+1})x_i\\
&-\sum_{i=1}^n\max\{x_{i-1}-x_i,0\}I\\
&-\sum_{i=1}^n\max\{x_i-x_{i-1}-1,0\}I\\
&-\sum_{i=k}^n\max\{m_s+x_{i-k}-x_i,0\}I\\
&-\sum_{i=k}^n\max\{m_e-k+x_i-x_{i-k},0\}I.
\end{aligned}
\]

注意到这个形式满足模型

\[\min\quad \sum_{u}b_ux_u+\sum_{\lang v,u\rang}c_{uv}\max\{0,x_v-x_u-w_{uv}\},
\]

因此可以直接翻译为费用流建图. 先取个负把 \(\max\) 变成 \(\min\), 令 \(b_i=s_{i+1}-s_i+e_i-e_{i+1}\), 流网络 \(G=(V,E)\) 建图如下:

\[\begin{aligned}
V &= \{S,T\}\cup\{1,2,\dots,n\},\\
E &= \{\lang S,i,b_i,0\rang\mid b_i>0\} \cup \{\lang u,T,-b_i,0\rang\mid b_i<0\}\\
&\cup \{\lang i,i-1,I,0\rang\}\cup\{\lang i-1,i,I,1\rang\}\\
&\cup \{\lang i,i-k,I,-m_s\rang\}\cup\{\lang i-k,i,I,k-m_e\rang\}.
\end{aligned}
\]

设最小费用为 \(c\), 对应的解集 \(\{x_{0..n}\}\). 注意此时可能有 \(x_0\neq0\), 所以需要令 \(x_i\gets x_i-x_0\) 并得到实际的最小费用 \(c'\), 则 \(\sum_{i=1}^ne_i+c'\) 就是答案, \(\{x_{0..n}\}\) 的差分就是方案. 复杂度为 \(\mathcal O(\text{Dinic}(n,5n))\).

  不过 ... 求这个解集 \(\{x_{0..n}\}\) 并不太容易. 神 crashed 称可以将残量网络上的 \(f_{uv}\) 对偶回原线规, 然后根据互松弛定理得到关于 \(x\) 的取等条件, 最后建差分约束解出 \(x\). 现在你知道我没有写代码的原因啦!

\(\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;
typedef std::pair<LL, int> PLI;
#define fi first
#define se second const int MAXN = 1e3;
int n, k, ms, me, s[MAXN + 5], e[MAXN + 5]; namespace FG { // #define POSITIVE_FLAG // determined if spfa is necessary. const int MAXND = MAXN + 1, MAXEG = MAXN * 2 + 1;
const LL LINF = 1ll << 60;
int S, T, ecnt = 1, head[MAXND + 5], curh[MAXND + 5];
LL dis[MAXND + 5], hgt[MAXND + 5];
struct Edge { int to, flw, cst, nxt; } graph[MAXEG * 2 + 5];
bool instk[MAXND + 5]; inline void link(const int s, const int t, const int f, const int c) {
// printf("%d %d %d,%d\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;
} #ifndef POSITIVE_FLAG
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]) inq[v] = true, que.push(v);
}
}
}
return dis[T] != LINF;
}
#endif inline bool dijkstra() {
static std::priority_queue<PLI, std::vector<PLI>, std::greater<PLI>> heap;
rep (i, S, T) hgt[i] += dis[i], dis[i] = LINF;
heap.emplace(dis[S] = 0, S);
while (!heap.empty()) {
PLI p(heap.top()); heap.pop();
if (p.fi != dis[p.se]) continue;
for (int i = head[p.se], v; i; i = graph[i].nxt) {
LL d = p.fi + graph[i].cst - hgt[v = graph[i].to] + hgt[p.se];
if (graph[i].flw && d < dis[v]) heap.emplace(dis[v] = d, v);
}
}
return dis[T] != LINF;
} inline PIL augment(const int u, int iflw) {
if (u == T) return { iflw, 0 };
PIL ret(0, 0); instk[u] = true;
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 - hgt[v] + hgt[u]) {
PIL t(augment(v, std::min(iflw, graph[i].flw)));
graph[i].flw -= t.fi, graph[i ^ 1].flw += t.fi;
ret.fi += t.fi, iflw -= t.fi;
ret.se += t.se + 1ll * graph[i].cst * t.fi;
if (!iflw) break;
}
}
if (ret.fi) instk[u] = false;
return ret;
} inline PIL dinic() {
PIL ret(0, 0);
#ifdef POSITIVE_FLAG
while (dijkstra()) {
#else
for (spfa(); dijkstra();) {
#endif
rep (i, S, T) curh[i] = head[i], instk[i] = false;
PIL t(augment(S, 0x3f3f3f3f));
ret.fi += t.fi, ret.se += t.se;
}
return ret;
} } // namespace FG. int main() {
scanf("%d %d %d %d", &n, &k, &ms, &me);
rep (i, 1, n) scanf("%d", &s[i]);
rep (i, 1, n) scanf("%d", &e[i]); FG::S = 0, FG::T = n + 1;
rep (i, 1, n) FG::link(i, std::min(i + k, n + 1), 1, s[i] - e[i]);
rep (i, 1, k - 1) FG::link(i, i + 1, k - ms, 0);
rep (i, k, n) FG::link(i, i + 1, k - ms - me, 0);
FG::link(FG::S, 1, k - ms, 0); LL ans = 0;
rep (i, 1, n) ans += s[i];
printf("%lld\n", ans - FG::dinic().se);
rep (i, 1, n) putchar("ES"[FG::graph[i << 1].flw]);
putchar('\n');
return 0;
}

Solution -「NEERC 2016」Delight for a Cat 的一个尝试的更多相关文章

  1. Solution -「ZJOI 2016」「洛谷 P3352」线段树

    \(\mathcal{Descrtiption}\)   给定 \(\{a_n\}\),现进行 \(m\) 次操作,每次操作随机一个区间 \([l,r]\),令其中元素全部变为区间最大值.对于每个 \ ...

  2. Solution -「CERC 2016」「洛谷 P3684」机棚障碍

    \(\mathcal{Description}\)   Link.   给一个 \(n\times n\) 的网格图,每个点是空格或障碍.\(q\) 次询问,每次给定两个坐标 \((r_1,c_1), ...

  3. Solution -「NOI 2016」「洛谷 P1587」循环之美

    \(\mathcal{Description}\)   Link.   给定 \(n,m,k\),求 \(x\in [1,n]\cap\mathbb N,y\in [1,m]\cap \mathbb ...

  4. Solution -「SDOI 2016」「洛谷 P4076」墙上的句子

    \(\mathcal{Description}\)   Link.   (概括得说不清话了还是去看原题吧 qwq. \(\mathcal{Solution}\)   首先剔除回文串--它们一定对答案产 ...

  5. Solution -「APIO 2016」「洛谷 P3643」划艇

    \(\mathcal{Description}\)   Link & 双倍经验.   给定 \(n\) 个区间 \([a_i,b_i)\)(注意原题是闭区间,这里只为方便后文描述),求 \(\ ...

  6. Solution -「ARC 104E」Random LIS

    \(\mathcal{Description}\)   Link.   给定整数序列 \(\{a_n\}\),对于整数序列 \(\{b_n\}\),\(b_i\) 在 \([1,a_i]\) 中等概率 ...

  7. Solution -「简单 DP」zxy 讲课记实

    魔法题位面级乱杀. 「JOISC 2020 Day4」治疗计划 因为是不太聪明的 Joker,我就从头开始理思路了.中途也会说一些和 DP 算法本身有关的杂谈,给自己的冗长题解找借口. 首先,治疗方案 ...

  8. Solution -「CTS 2019」「洛谷 P5404」氪金手游

    \(\mathcal{Description}\)   Link.   有 \(n\) 张卡牌,第 \(i\) 张的权值 \(w_i\in\{1,2,3\}\),且取值为 \(k\) 的概率正比于 \ ...

  9. Solution -「BZOJ 3812」主旋律

    \(\mathcal{Description}\)   Link.   给定含 \(n\) 个点 \(m\) 条边的简单有向图 \(G=(V,E)\),求 \(H=(V,E'\subseteq E)\ ...

  10. Solution -「CF 1342E」Placing Rooks

    \(\mathcal{Description}\)   Link.   在一个 \(n\times n\) 的国际象棋棋盘上摆 \(n\) 个车,求满足: 所有格子都可以被攻击到. 恰好存在 \(k\ ...

随机推荐

  1. Python面向对象小备忘

    最近学到面向对象了,感觉到Python这方面的语法也有点神奇,这里专门归纳一下Python面向对象中我觉得比较重要的笔记. 本文目前有的内容:实例属性和类属性的访问,使用@property修饰器 实例 ...

  2. 2024 Navicat Premium 16+17安装教程(附激活方法)

    Navicat Premium,作为一款功能全面的集成数据库管理工具,无缝支持多样化的数据库类型,为用户带来前所未有的高效与便捷管理体验.它不仅涵盖了连接管理.数据导入导出.同步迁移.备份恢复等核心功 ...

  3. .NET 创建动态方法方案及 Natasha V9

    前言 本篇文章前面客观评估了 .NET 创建动态方方案多个方面的优劣,后半部分是 Natasha V9 的新版特性. .NET 中创建动态方法的方案 创建动态方法的不同选择 以下陈列了几种创建动态方法 ...

  4. gearman实现任务分发

    偶然发现了这个gearman,觉得这玩意儿挺好用,非常适合PHP运行一部分业务. 话不多说,安装一下. 使用apt查找 sudo apt search gearman 找到了这个 gearman/bi ...

  5. (Python基础教程之十一)Python找到最大的N个(前N个)或最小的N个项目

    Python基础教程 在SublimeEditor中配置Python环境 Python代码中添加注释 Python中的变量的使用 Python中的数据类型 Python中的关键字 Python字符串操 ...

  6. 从几个sample来学习Java堆,方法区,Java栈和本地方法栈

    最近在看<深入理解Java虚拟机>,书中给了几个例子,比较好的说明了几种OOM(OutOfMemory)产生的过程,大部分的程序员在写程序时不会太关注Java运行时数据区域的结构: 感觉有 ...

  7. Docker之磁盘清理

    Docker 很占用空间,每当我们运行容器.拉取镜像.部署应用.构建自己的镜像时,我们的磁盘空间会被大量占用. 如果你也被这个问题所困扰,咱们就一起看一下 Docker 是如何使用磁盘空间的,以及如何 ...

  8. nginx之日志

    1)耗时问题定位 这几天在优化服务器的响应时间,在根据 nginx 的 accesslog 中 requesttime进行程序优化时,发现有个接口,直接返回数据,平均的requesttime进行程序优 ...

  9. 从零开始学java(前言)

    很长时间以来,自己都非常的懒惰,以前学习的知识都已经记不大清了,入职半年以来,学到的东西不多,反倒以前会的都不会认识了 打算从新开始,从心出发,认真学习 注册博客园是看了狂神说的建议,希望通过养成每天 ...

  10. spring 使用异步任务

    1.说明 在springboot 中使用 @Async 实现异步任务处理,下面介绍一下如何实现这个. 2.实现代码 2.1 增加@EnableAsync @EnableAsync public cla ...