这题真的想了挺久的,然而到最后也还是没想到怎样处理环的情况……网上竟然也完全没有题解,无奈之下到 CF 的 AC 代码里面去找了一份膜拜了一下。感谢~

  由于觉得这题有一定的难度,自己看代码也看了比较久才理解,就记录一下吧……

  首先,不难发现一个人一定不会往返走。因为一个人之所以要往回走,理由一定是因为碰到了已经访问过的凳子(与他人的路径相撞)。我们不妨看作是这两个人灵魂交换,即两人均继续向前,而不往返。所以问题转化为在一个环上指定 n 个点,从每个点可以顺 \ 逆时针延伸出一条长为 x 的线段,问能覆盖整个环的最小 x 值。

  显然需要二分一下。如果这个问题是在序列上的问题,我们可以 dp。设立 dp 状态 f[i][0 / 1] 表示第 i 号节点是向左 / 向右的线段,所能从起点开始连续访问的最远地点是多少。这样好像有个 bug :可以有下图中的情况出现,即第 i - 1 到第 i 号节点之间的位置由第 i + 1 号节点来覆盖。但这样没有关系,因为我们可以在 dp 到 i + 1 的位置的时候填补起 i ~ i - 1 之间的位置,并把答案更新为第 i 号节点向前延伸出来的部分。

  如果不够清晰,可以看一下代码中的转移方程。

  最难处理的就是这一个环。因为有可能第 1 号节点和第 2 号节点之间的这一段是由第 n 号节点来覆盖的。可是这样根据我们的定义,我们将一直无法更新 dp 数组。为了避免这样的一种情况,我们可以选择破环为链。断开哪一条边呢?选择断开最长的一条边,因为这条边就是答案的上界,也就是指这样断开的链上 n 号节点永远不会选择通过这条边来覆盖 1~2 之间的这一段。然后我们就只需要分类讨论一下 1 号节点是向左还是向右就可以了。

  **启示:其实这种破环为链的题目见得很多了,但还是没有做出来,是我太菜了……应该找到在环上必然不会有影响的一条边去分割成链,如51nod部落信号。

  代码(kuai的...):

#include <bits/stdc++.h>
using namespace std;
#define maxn 1000000
#define int long long
#define INF 99999999999LL
int n, m, f[maxn][], P[maxn], p[maxn];
int ans, mx, pos; int read()
{
int x = , k = ;
char c; c = getchar();
while(c < '' || c > '') { if(c == '-') k = -; c = getchar(); }
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x * k;
} bool Check(int x)
{
for(int j = ; j <= ; j ++)
{
f[][] = j ? -INF : ; f[][] = j ? x : -INF;
if(j)
{
if(x + >= P[]) f[][] = max(P[] + x, P[]);
else f[][] = -INF;
}
for(int i = j + ; i <= n; i ++)
{
f[i][] = f[i][] = max(f[i - ][], f[i - ][]);
if(max(f[i - ][], f[i - ][]) + >= P[i]) f[i][] = P[i] + x;
if(P[i] - x - <= f[i - ][]) f[i][] = max(f[i - ][], P[i]);
if(P[i] - x - <= f[i - ][]) f[i][] = max(f[i - ][], max(P[i], P[i - ] + x));
}
int t = max(f[n][], f[n][]);
if(j && t + >= P[] + m - x) return ;
else if(!j && t + >= m - x) return ;
}
return ;
} signed main()
{
m = read(), n = read();
if(n == ) { printf("%I64d\n", m - ); return ; }
for(int i = ; i <= n; i ++) p[i] = read();
for(int i = ; i <= n; i ++)
{
int dis; dis = p[i] - p[i - ];
if(i == ) dis = p[i] + m - p[n];
if(dis > mx) mx = dis, pos = i;
} for(int i = ; i <= n; i ++)
{
int t = i - + pos; if(t > n) t -= n;
P[i] = p[t]; P[i] -= p[pos];
if(P[i] < ) P[i] += m;
} int l = , r = m - ;
while(l <= r)
{
int mid = (l + r) >> ;
if(Check(mid)) ans = mid, r = mid - ;
else l = mid + ;
}
printf("%I64d\n", ans);
return ;
}

【题解】CF#713 E-Sonya Partymaker的更多相关文章

  1. 【CodeForces】713 C. Sonya and Problem Wihtout a Legend

    [题目]C. Sonya and Problem Wihtout a Legend [题意]给定n个数字,每次操作可以对一个数字±1,求最少操作次数使数列递增.n<=10^5. [算法]动态规划 ...

  2. 竞赛题解 - CF Round #524 Div.2

    CF Round #524 Div.2 - 竞赛题解 不容易CF有一场下午的比赛,开心的和一个神犇一起报了名 被虐爆--前两题水过去,第三题卡了好久,第四题毫无头绪QwQ Codeforces 传送门 ...

  3. Codeforces 713 C Sonya and Problem Wihtout a Legend

    Description Sonya was unable to think of a story for this problem, so here comes the formal descript ...

  4. 题解——CF Manthan, Codefest 18 (rated, Div. 1 + Div. 2) T5(思维)

    还是dfs? 好像自己写的有锅 过不去 看了题解修改了才过qwq #include <cstdio> #include <algorithm> #include <cst ...

  5. 竞赛题解 - [CF 1080D]Olya and magical square

    Olya and magical square - 竞赛题解 借鉴了一下神犇tly的博客QwQ(还是打一下广告) 终于弄懂了 Codeforces 传送门 『题目』(直接上翻译了) 给一个边长为 \( ...

  6. [题解] [CF 1250J] The Parade

    题面 题目大意: 给定一个 \(n\) , 所有军人的数量均在 \([1, n]\) 给定 \(a_i\) 代表高度为 \(i\) 的军人的个数 你要将这些军人分成 \(k\) 行, 满足下面两个条件 ...

  7. 题解 CF 1372 B

    题目 传送门 题意 给出 \(n\),输出 \(a\) ,\(b\) (\(0 < a \leq b < n\)),使\(a+b=n\)且 \(\operatorname{lcm}(a,b ...

  8. 题解——CF Manthan, Codefest 18 (rated, Div. 1 + Div. 2) T4(模拟)

    随便模拟下就过了qwq 然后忘了特判WA了QwQ #include <cstdio> #include <algorithm> #include <cstring> ...

  9. 题解——CF Manthan, Codefest 18 (rated, Div. 1 + Div. 2) T3(贪心)

    是一道水题 虽然看起来像是DP,但其实是贪心 扫一遍就A了 QwQ #include <cstdio> #include <algorithm> #include <cs ...

随机推荐

  1. CF 96 D. Volleyball

    D. Volleyball http://codeforces.com/contest/96/problem/D 题意: n个路口,m条双向路,每条长度为w.每个路口有一个出租车司机,最多可以乘坐这辆 ...

  2. vmware因为软件出过一次复制的错误导致不能复制到主机的解决方法

    只需要把vmware的虚拟机进程全部结束掉,然后重置(先设置不勾选复制等,然后保存后在勾选上并保存)一次虚拟机隔离设置(需要在关闭虚拟机的情况下设置,否则就是灰色不允许操作),然后再开启虚拟机,就能正 ...

  3. Egret入门(三)--创建HelloWorld项目(4.0-使用Egret Wing)

    准备 编辑器: Egret Wing3(4.0.3) 需要下载安装 语言: TepyScript(JS的超集,参考手册http://bbs.egret.com/thread-1441-1-1.html ...

  4. Unity编辑器扩展 Chapter3--Create Custom Inspector

    一.Create Custom Inspector 重绘inspector面板一方面是我们的挂在脚本的窗口变得友好,另一方面可以让其变得更强大,比如添加一些有效性验证. 二.重要说明 1.Editor ...

  5. PLSQL事务

    1 使用set transaction设置事务属性 2 只读事务 set transaction read only 3 读写事务 set transaction write; 4 在进行数据统计分析 ...

  6. http 502 bad gate way

    世界杯期间,公司的cdn在回源时突然出现大量502. 刚出现问题时,因为考虑到一般502都是上游服务器出现问题,然后因为已经服务了很久都没有出现过这种问题, 就没有仔细考虑,就让回源中心的同事进行排查 ...

  7. 解决登录linux输入密码问题

    1.使用密钥 ssh-keyssh -i .ssh/*.key root@<ip_addr> 2.使用sshpass 安装 rpm 包:yum install sshpass 配置文件: ...

  8. react native基础与入门

    react native基础与入门 一.react native 的优点 1.跨平台(一才两用) 2.低投入高回报 (开发成本低.代码复用率高) 3.性能高:拥有独立的js渲染引擎,比传统的h5+ w ...

  9. Python Requests库入门——应用实例-百度、360搜索关键词提交

    百度的关键词接口: http://www.baidu.com/s?wd=keyword 360的关键词接口: http://www.so.com/s?q=keyword keyword就是需要查找的关 ...

  10. 【探路者】Alpha发布用户使用报告

    预期统计用户使用数量:13人. 博文内容:1用户列表.2评论列表.3统计与总结 1用户列表: 二.评论内容 用户1:1不够好看.2不应该是中国地图为背景,蛇头是人物头像的么?(那是宣传片,不是预览图) ...