我感觉这个题作为Day2T1,有一定的挑战性。为$Rxd$没有完成这道题可惜。

我觉得这道题,如果按照前几个部分分的思路来想,就有可能绕进错误的思路中。因为比如说每个传感器最多只在序列中出现$2$次,很有可能会想到分别在每一个传感器之后用开关来控制。我在做这个题的时候就因为这个思路陷入僵局。事实上这个做法的弊端十分明显,首先如果可行的话,实现起来也会十分复杂,毕竟所有传感器序列中交错复杂,难于处理;其次,对于每一个传感器分别使用开关很可能导致开关的浪费。

这道题的两个关键的想法都源于其中的两个部分分,部分分的设置不仅是得分,往往对标算也有一定的启发作用。

把所有传感器的出口汇合在一起:

第$4$个$Subtask$中$n = 16$,这个$2$的幂次的数提示我们可以改变思路,把序列整体起来考虑可能更加简洁,更加清晰。

这么做会让题目简化,我们可以把那个汇集了所有传感器的开关向下建成一棵满二叉树(这里说明一下为什么每层都是满的,因为要保证每个开关都被经过了偶数次),底层$O(n)$个节点可以根据访问到的顺序确定它该连向那个传感器,或者连回开关的根。底层的节点数$L$应该是大于等于$\frac{n + 1}{2}$的最小的一个$2$的幂次。总的开关数就是$2L - 1$。在第$4$个$Subtask$中,我们按照这个方法只需要$15$个开关就能解决问题。

这里有一个小问题,就是如何知道某一个底层的出口会被第几次访问到。观察它的性质,发现它与$Fft$中的$rev$非常像,也就是从左到右第$i$个出口会被第$rev_{i}$次访问到。

于是看似这个算法很完美(事实上它已经十分优秀了,你可以光在后三个$Subtask$中得到$47$分),却有一点小瑕疵。当$n$是一个$2$的幂次多一点点的时候,$L$最坏会达到$n$个,也就是总节点数会被卡到$2n$个。这是一个很明显有很痛苦的瑕疵,所幸我们还有解决方法。

把冗余的开关剔除:

考虑到如果开关的某个出口没有被利用,我们之前的解决方法中,我们将它连向了开关的根,这是一次无用的滚动。我们唯一的问题就在于无用的滚动可能会过多。一个简单的想法就是如果某一个开关的两个出口都是连向开关的根的,那显然这个开关是无用的,我们可以把他直接去掉,把它父亲的出口连到开关的根去。如此做,直到没有什么可以删的点为止。注意,最后一次访问到的底层开关不能删除,因为要保证每一个开关都被经过了偶数次。

可惜直接这么做效果微乎其微,我们发现问题在于,如果直接把第$i$次访问到的底层的出口连向序列的第$i$个传感器,我们利用到的开关在底层中排布的是非常凌乱的,其实具体选那些出口没有什么太大关系,只要保证球滚出的顺序和要求的序列一样就可以了。我们可以只用$L$个底层出口中的后面$n$个,这么做就可以把所有利用到的开关分在一起,至于那些无用的开关就会被集中处理。这样节点的数量就会大大减少,总共有$logn$层开关,每层开关最多只有一个开关是不满的,所以不满的开关数最多$logn$个,由于满二叉树的节点数不会超过$n$个,故节点的数量就会在$n + logn$以内了。

$\bigodot$技巧&套路:

  • 部分分对思路的禁锢和启发
  • 线段树的思想,$Fft$的$rev$的意义
#include "doll.h"
#include <vector>
#include <algorithm> using namespace std; const int N = ; int n, L, tot = ;
int rev[N], bit[N], ps[N];
vector<int> a, c, x, y, to; int Solve(int l, int r) {
if (l == r) return (l >= L - n)? a[ps[l]] : N;
int md = (l + r) >> ;
int lc = Solve(l, md);
int rc = Solve(md + , r);
if (lc == N && rc == N) return N;
x.push_back(lc);
y.push_back(rc);
return -x.size();
} void create_circuit(int m, vector<int> a0) {
a = a0, n = a.size();
a.push_back();
for (L = ; L < n; L <<= );
for (int i = ; i < L; ++i)
rev[i] = (rev[i >> ] >> ) | (i & ? L >> : );
fill(bit, bit + L, -);
for (int i = L - n; i < L; ++i) bit[rev[i]] = i;
for (int i = ; i < L; ++i)
if (bit[i] != -) to.push_back(bit[i]);
for (int i = ; i < to.size(); ++i) ps[to[i]] = i + ;
int rt = Solve(, L - );
c.resize(m + , rt);
c[] = a[];
for (int &i : x) if (i == N) i = rt;
for (int &i : y) if (i == N) i = rt;
answer(c, x, y);
}

【IOI 2018】Doll 机械娃娃的更多相关文章

  1. 【IOI 2018】Werewolf 狼人

    虽然作为IOI的Day1T3,但其实不是一道很难的题,或者说这道题其实比较套路吧. 接下来讲解一下这个题的做法: 如果你做过NOI 2018的Day1T1,并且看懂了题面,那你很快就会联想到这道题,因 ...

  2. [IOI2018]机械娃娃——线段树+构造

    题目链接: IOI2018doll 题目大意:有一个起点和$m$个触发器,给出一个长度为$n$的序列$a$,要求从起点出发按$a$的顺序经过触发器并回到起点(一个触发器可能被经过多次也可能不被经过), ...

  3. 【IOI 2018】Combo 组合动作(模拟,小技巧)

    题目链接 IOI的签到题感觉比NOI的签到题要简单啊,至少NOI同步赛我没有签到成功…… 其实这个题还是挺妙妙的,如果能够从题目出发,利用好限制,应该是可以想到的做法的. 接下来开始讲解具体的做法: ...

  4. 【IOI2018】机械娃娃

    看到的时候感到很不可做,因为所有的开关都要状态归零.因此可以得到两分的好成绩. --然后 yhx-12243 说:这不是线段树优化建图吗? 于是我获得了启发,会做了-- 还不是和上次一样,通过提示做出 ...

  5. 【IOI 2018】Highway 高速公路收费

    这是一道极好的图论题,虽然我一开始只会做$18$分,后来会做$51$分,看着题解想了好久才会做(吐槽官方题解:永远只有一句话),但这的确是一道好题,值得思考,也能启发思维. 如果要讲这道题,还是要从部 ...

  6. [IOI 2018] Werewolf

    [题目链接] https://www.luogu.org/problemnew/show/P4899 [算法]         建出原图的最小/最大生成树的kruskal重构树然后二维数点 时间复杂度 ...

  7. Before NOIP 2018

    目录 总结 刷题 2018 - 9 - 24 2018 - 9 - 25 2018 - 9 - 26 2018 - 9 - 27 2018 - 9 - 28 2018 - 9 - 29 2018 - ...

  8. WC2019 题目集

    最近写的一些 WC2019 上讲的一些题.还是怕忘了,写点东西记录一下. LOJ2983 「WC2019」数树 题意 本题包含三个问题: 问题 0:已知两棵 \(n\) 个节点的树的形态(两棵树的节点 ...

  9. WC2019游记

    本来不打算写游记的,但后来想了想这么一次难忘的经历总该留下点痕迹吧...... DAY-1 走之前的最后一天,因为前一天晚上打了CF,所以早上9点才到机房.写了一道圆方树深深地体会到了来自仙人掌的恶意 ...

随机推荐

  1. OpenGL学习笔记(4) GLM库的使用

    OpenGL和DirextX不一样,没有内置的数学库,于是我们需要找一个第三方库,按照LearnOpenGL的教程我们使用GLM库,可以到他们的官网下载 glm常用的数据类型 vec2 二维向量 ve ...

  2. Codeblocks自动代码格式化快捷键(自带)

    代码区域右击 点format use AStyle 估计也就是考试竞赛逼着用这个

  3. php学习--变量和数据类型

    PHP变量 变量 ​ 程序执行期间,可以变化的量即为变量. 声明变量 以美元$ 符号声明 注意:(PHP严格区分大小写) 变量名称以 字母.或下划线开始,后面跟上数字/字母/下划线,不能包含特殊字符 ...

  4. Python处理PDF和Word文档常用的方法(二)

    Python处理word时,需要安装和导入python-docx模块. 安装命令:pip install python-docx 导入命令:import docx 编码编写顺序:用docx.Docum ...

  5. pssh命令详解

    基础命令学习目录首页 原文链接:https://www.cnblogs.com/kevingrace/p/6378719.html pssh提供OpenSSH和相关工具的并行版本.包括pssh,psc ...

  6. 随手记录-linux-Shellinabox插件

    Shellinabox 是一个利用 Ajax 技术构建的基于 Web 的远程Terminal 模拟器,也就是说安装了该软件之后,不需要开启 ssh服务,通过 Web 网页就可以对远程主机进行维护操作了 ...

  7. “Hello World!“”团队第七周召开的第二次会议

    今天是我们团队“Hello World!”团队第七周召开的第二次会议.博客内容: 一.会议时间 二.会议地点 三.会议成员 四.会议内容 五.todo list 六.会议照片 七.燃尽图 八.代码 一 ...

  8. mybatis之模糊查询SQL

    一,MySQL数据库 name like concat('%' , #{name} , '%') 二,Oracle数据库 name like '%' || #{name} || '%'

  9. 冲刺One之站立会议4 /2015-5-17

    今天我们继续了昨天未完成的部分,把服务器端的在线人数显示做了出来,但是在调试的时候还有一些不可预知的自己也不会改的bug,让我们有点不知所措,启动时间的显示相对来说比较容易实现. 燃尽图4

  10. spring冲刺第九天

    昨天使炸弹可以炸死人物并可以炸没砖块,并试着将小怪加入地图. 今天设计游戏的道具,比如吃了道具人物反方向运动等. 遇到的问题设计不够完善,道具单一.