看到的时候感到很不可做,因为所有的开关都要状态归零。因此可以得到两分的好成绩。

……然后 yhx-12243 说:这不是线段树优化建图吗?

于是我获得了启发,会做了……

还不是和上次一样,通过提示做出这种交互题的?

我还是太菜了

以下魔改自我的思考过程(一开始想对每一个触发器配一组开关决策下一步,然后听说用线段树直接想到一组开关决策整颗树了)

其实前半部分的思考貌似没用,直接套到整棵树上是适用的


因为状态要清零,考虑清空这个状态。

考虑每个触发器连向一个开关,来决策下一步走什么,因为和二进制有关,所以我们构造一个线段树类似的。

假设出现 \(s\) 次,记 \(L = 2^k \geq s\) 使得 \(k\) 最小。

那么我们前 \(L - s\) 次就开个到根的边,后面 \(s\) 次就可以直接走后继了。

那么如果走了 \(L\) 步,线段树内所有点都遍历到了。

注意到 \(L\) 是二的幂,跑完后所有非叶节点区间长度都是偶数,即遍历到偶数次,即清零了。这也说明了为什么不用普通线段树。

所以要把叶子结点全部扔掉,不然就会有节点只遍历到一次。如果扔掉叶子,那么就是一棵大小为 \(L - 1\) 的树。

注意特判一下最后一个点的后继是 \(0\)。

这样点数是 \(2n\) 的 (对于一个 \(s\),有 \(s \leq L < 2s\)),可以获得 \(50\) 分。

考虑优化这个,发现很多的点是没有用的,即整个区间都会再次回到根。那么要访问到这个区间的时候,不如提前回到根。即删掉这个点,父亲连向自己的边改为连向根。

但是发现优化力度不大,这是因为这道题的遍历方式使得遍历的点交错了起来。

因为最后一个点是在右边的,那么我们把线段树右边的那 \(s\) 个点取来,钦定它们是出边,然后根据先前遍历的顺序把值钦定一下就好。

我们只选了右边的点,优化力度很明显了。

对于 \(s = 5\), \(L = 8\),淡色是叶子,打钩的是有用的点。(机房没带数位板请见谅,其他软件不会用,求dalao推荐)

可以证明对于一个 \(N\) ,点数是 \(N - 1 + \lceil \log_2 N \rceil\)。考虑对于相同的 \(L\), 考虑 \(N\) 个叶子,至少要 \(N - 1\) 个点,而多出来的是单独一条链,也就是最后一个元素向上的链。这个链长度小于等于 \(\lceil \log_2 N \rceil\)。

所以这样子就能得到较为优秀的分数了。

然而对于每一个触发器,我们都会多一个 \(\log\)。那么为什么不放在一起呢?这样正好符合正解的复杂度。

考虑用一个统一的开关线段树维护这些触发器,然后触发器回到决策线段树的开头。

注意触发器不能回到起点,因为一个触发器可能出现多次,那么我们很难决定它是最后一个的时候怎么做。

为了回到起点,我们增加一个虚点,或者换句话说,就是往规定的触发器序列末尾加一个 \(0\)。

和上面的方法类似,我们把叶子的遍历序列存下来,同时只取在线段树右边的 \(N + 1\) 个点,根据遍历的时间来钦定触发器。

一样的,因为遍历了二的幂次,所以所有开关归零,经过开关次数 \(O\left(N \log N\right)\) 大概 \(3 \times 10^6\),也不会爆炸。

开关个数 \(O\left(N + \lceil \log_2 \left(N + 1\right) \rceil\right)\),因为judger对这个 \(\log\) 的界卡的不紧,大概可以大 \(1\),所以刚好能通过此题。

看到有些人用bitreverse,的确这样子就是bitreverse后的不断+1,可以给各种操作带来方便。

  1. #include "doll.h"
  2. #include <bits/stdc++.h>
  3. typedef std::vector<int> VI;
  4. const int MAXN = 262145;
  5. const int spe = 19260817;
  6. int idx[MAXN], xs[MAXN], ys[MAXN];
  7. int qli[MAXN];
  8. int typ[MAXN], cnto[MAXN], tot, lim, N;
  9. int tli[MAXN], t0t, cnt[MAXN];
  10. void qry(int u, int l, int r) {
  11. if (l == r) {
  12. cnto[l] = ++tot;
  13. return ;
  14. }
  15. int mid = l + r >> 1;
  16. !typ[u] ? qry(u << 1, l, mid) : qry(u << 1 | 1, mid + 1, r);
  17. typ[u] ^= 1;
  18. }
  19. void relable(int u, int l, int r) {
  20. if (l == r) {
  21. tli[++t0t] = l;
  22. return ;
  23. }
  24. int mid = l + r >> 1;
  25. relable(u << 1 | 1, mid + 1, r);
  26. relable(u << 1, l, mid);
  27. }
  28. int txt;
  29. int build(int l, int r) {
  30. if (l == r) {
  31. int now = cnt[l];
  32. return now > 0 ? qli[now] : spe;
  33. }
  34. int mid = l + r >> 1;
  35. int tl = build(l, mid), tr = build(mid + 1, r);
  36. if (tl == spe && tr == spe) return spe;
  37. int now = ++txt;
  38. xs[now] = tl, ys[now] = tr;
  39. return -now;
  40. }
  41. void create_circuit(int M, VI A) {
  42. N = A.size(); ++N;
  43. lim = 1;
  44. while (lim < N) lim <<= 1;
  45. for (int i = 1; i <= lim; ++i) qry(1, 1, lim);
  46. relable(1, 1, lim);
  47. std::sort(tli + 1, tli + 1 + N, [] (int a, int b) { return cnto[a] < cnto[b]; });
  48. for (int i = 1; i <= N; ++i)
  49. cnt[tli[i]] = i;
  50. std::copy(A.begin(), A.end(), qli + 1);
  51. VI C(M + 1, 0), X, Y;
  52. int rt = build(1, lim);
  53. for (int i = 1; i <= txt; ++i) {
  54. xs[i] == spe ? xs[i] = rt : 0;
  55. ys[i] == spe ? ys[i] = rt : 0;
  56. }
  57. for (int i = 0; i <= M; ++i) C[i] = rt;
  58. X.assign(xs + 1, xs + 1 + txt);
  59. Y.assign(ys + 1, ys + 1 + txt);
  60. answer(C, X, Y);
  61. }

【IOI2018】机械娃娃的更多相关文章

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

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

  2. 【IOI 2018】Doll 机械娃娃

    我感觉这个题作为Day2T1,有一定的挑战性.为$Rxd$没有完成这道题可惜. 我觉得这道题,如果按照前几个部分分的思路来想,就有可能绕进错误的思路中.因为比如说每个传感器最多只在序列中出现$2$次, ...

  3. WC2019游记

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

  4. WC2019 题目集

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

  5. WC 2019 游记 - 败者之低语

    败者之低语 WC 2019 游记 Day -1 看了一圈PKU和THU的题,感觉图像识别真有意思... 感觉非常讲道理,pku还是比thu简单一点的... 听说高二414在thu没有进面试? 震惊!( ...

  6. [LeetCode] Russian Doll Envelopes 俄罗斯娃娃信封

    You have a number of envelopes with widths and heights given as a pair of integers (w, h). One envel ...

  7. 入手Invicta 8926 OB潜水自动机械腕表

    前个月前就想入手一款手表了,之前在关注和学习.询问他人选哪样的表好,前些天还在看精工Seiko机械表系列,今凌晨有朋友给我推荐这款Invicta 8926系列手表,我一看便喜欢了. 在网上也是搜索了很 ...

  8. 固态硬盘与机械硬盘 SQL Server 单表插入性能对比测试

    测试环境

  9. 机械革命 USB装系统各种坑

    买了个号称超强性价比的游戏本- 机械革命, i7+ssd+hd+4G RAM+ GTX850M, 很直接, 直接出厂就一个DOS系统,回来要自己装机. 好吧, 先下了个大白菜软件,用来刻录ISO系统到 ...

随机推荐

  1. T100——动态更改Label的说明

    例子: #設定科目名稱    IF g_prog = 'aapt300' THEN       CALL cl_set_comp_att_text("lbl_apca036",cl ...

  2. 阿里云自动获取token值(python)

    一,token说明 token的意思是“令牌”,是服务端生成的一串字符串,作为客户端进行请求的一个标识.当用户第一次登录后,服务器生成一个token并将此token返回给客户端,以后客户端只需带上这个 ...

  3. c# winfrom 界面设计

    1.在用DotnetBar的RibbonControl时,界面最大化时,会把电脑桌面的任务栏遮盖住: 解决办法:在load事件中写入: , Screen.PrimaryScreen.WorkingAr ...

  4. 【原创】大叔经验分享(67)spring boot启动报错

    spring boot 启动报错: Caused by: java.lang.IllegalArgumentException: LoggerFactory is not a Logback Logg ...

  5. global.css

    global.css /* 页面元素初始化和常用样式定义-start */ /*======== 全局 ========*/ body, div, dl, dt, dd, ul, ol, li, h1 ...

  6. (转)Android刷机的一些知识整理

    刷机概述刷机原因刷机可以升级和破解固件(在Android上:即可以升级系统,更改系统,获取Root权限):破解系统的原因①安装第三方软件不需要签名,不受证书的束缚:②修改系统的文件,达到系统的瘦身,以 ...

  7. 1.移动端自动化测试-知识笔记(面试必备,测试点,adb命令)

    移动端测试: 移动应用,特性(功能) 满足 需求(产品文档,隐性需求) 一.App功能测试: 死活背下来1.业务逻辑正确性测试: 产品文档,隐性需求- 写成测试用例 2.兼容性测试:   1.系统版本 ...

  8. 经典i++和++i问题(附带运算符优先级问题)

    转自 https://blog.csdn.net/mustard1020/article/details/79617865 1.i++和++i的区别     (1)i++简单来说就是先用i的值来参加表 ...

  9. SQL语句复习【专题三】

    SQL语句复习[专题三] DML 数据操作语言[insert into update delete]创建表 简单的方式[使用查询的结果集来创建一张表]create table temp as sele ...

  10. Makefile 编译静态库文件及链接静态库

    本文为原创文章,转载需指明该文链接 1.代码目录结构如下: comm/ comm/inc/apue.h  3 atexit.c Makefile  5 staticlib/lib/ staticlib ...