逻辑解构 | 秩序的帽子戏法

题目 数学大臣尤泽尔被国王亚路开德召到皇宫,却发现前面已经排了 9 个非常奇怪的人——这些人全都戴上了帽子,或黑或白。 早在一旁等候的侍卫克里亚,向一脸疑惑的尤泽尔走了过来。 “陛下正在让他们完成一个挑战”,克里亚解释说,“每次我都会给他们每个人随机戴上黑或白色的帽子,所有人要轮流猜自己头顶上的帽子的颜色,最多只可以猜错一次。” “原来这些帽子不是他们自己戴上的?但陛下恐怕要失望了,很显然人类无法看到自己的头顶”,尤泽尔略加思索,“除非里面有数学大臣。” “数学大臣就可以看到头顶吗?” 尤泽尔加入了挑战。 他们有一次商讨策略的机会,可以按任意顺序排队和报答案,但在排好队之后尤泽尔才会开始给他们戴帽子,排好队后只有嘴可以动。请问他们有完成挑战的方法吗? 答案(不唯一) 由于每个人都看得到前面的人的帽子,也听得到其他人报答案,所以以下策略是可行的: 每个人都面朝前方排成一条直线,并从后往前报答案 对于最后一个人,若前面有奇数顶黑帽子,就报黑色,若有偶数顶黑帽子,就报白色 对于其他人,根据后面的人报的颜色可以推算出自己的颜色 证明 由于每个人都看得到前面的人的帽子,也听得到其他人报答案,所以除了最后一个人以外,其他人: 可以看到前面有多少顶黑帽子 可以听到除了最后一个人,后面有多少顶黑帽子 最后一个人报出了前 9 个人共有奇数还是偶数顶黑帽子 不妨设总共有偶数顶黑帽子,否则就将颜色取反,此时有: 若前面有偶数顶,后面有偶数顶,那么自己一定是白帽子 若前面有偶数顶,后面有奇数顶,那么自己一定是黑帽子 若前面有奇数顶,后面有偶数顶,那么自己一定是黑帽子 若前面有奇数顶,后面有奇数顶,那么自己一定是白帽子 上述推理很好证明,以第一点为例:若前面有偶数顶,后面有偶数顶,如果自己还是黑帽子的话就总共有“前面偶数顶+后面偶数顶+自己1顶=奇数顶”了,与总共有偶数顶黑帽子矛盾,所以自己一定是白帽子。 其他三点证明也同理。因此该策略可以保证除了最后一个人以外必定猜对,可以完成挑战。 原理 原理在证明中已经体现得淋漓尽致了,就是奇偶性,然后舍弃最后一个人来传递信息,再充分利用游戏规则中隐含的已知信息来推算答案。 更深层的原理其实类似于本系列第一题的结论:在所有数值域均为模数的情况下,模意义下减法具有唯一解。 我们可以这么表述这个答案:设 $c_i$ 为从前往后第 $i$ 个人帽子的颜色,$0$ 为白色,$1$ 为黑色,这样从后往前报颜色,那么第 $i$ 个人: 可以看到前面的颜色和:$s_1=c_1+c_2+\cdots+c_{i-1}$ 可以听到后面的颜色和:$s_2=c_{i+1}+c_{i+2}+\cdots+c_9$ 最后一个人报了总颜色和:$s=c_1+c_2+\cdots+c_9$ 因此自然可以算出自己的颜色 $c_i=s-s_1-s_2$。上式全部是模 $2$ 意义下的,根据上述结论,$c_i$ 有唯一解。 当然,我们其实也可以将上述操作看作异或,这是另一种解释。 思考 可以更多人吗? 显然可以,因为上述策略总是可以保证除了最后一个人以外全对,因此无论有多少个人都可以完成挑战。 可以更多颜色吗? 也可以,因为该结论在任意模数下都是成立的,因此这个策略可以推广到任意颜色数 其他的答案? 在理解以上答案之后,我们不禁思考能否选择观察别的数据或别的性质。 毫无疑问,由于最后一个人已知信息最多,但唯独绝对不可能知道自己的信息,因此最后一个人一定是负责报一个全局信息来让其他人能够猜出自己颜色的。 什么样的信息符合要求呢?思考上述过程,我们发现要求是:对于任意已有的颜色序列,再加入不同的颜色会对信息造成不同的变化(不变也算作一种变化),且这一运算是存在逆运算的,且这一信息的可能状态数不超过颜色个数。 先只考虑两个颜色的情况,此时我们需要用一个 bit 来表示出包含所有人颜色在内的信息。 我绞尽脑汁想出来一个“长度为奇数的黑色段的奇偶性”,这是可行的,但其本质上其实与之前一个答案是一样的,因为黑帽子总数的奇偶性就取决于长度为奇数的黑色段的奇偶性,因此实际上是换一个表述。 还有其他的表述形式,例如“黑色与白色个数的差值(若 $n$ 为奇数还需要减去 $1$)是否是 $4$ 的倍数”、“黑色连续段的长度的异或和的奇偶性”,这些其实都等价于黑帽子个数的奇偶性。 实际上,对于每个人都是已知除了自己与最后一个人以外的所有颜色信息,所以我们只需要满足,在其他信息相同的情况下,自己是黑是白会导致全局信息不同即可。 用数学来描述的话,这样的全局信息必须对应一个函数 $f: {B,W}^{n-1} \to {0,1}$,使得对于每个位置 $i$ 和每个其他帽子的配置,改变第 $i$ 顶帽子的颜色都会改变函数值 $f$。 满足这种条件的函数,只能是前 $n-1$ 顶帽子黑帽子个数的奇偶性函数或其否定(即奇偶性的补)。 因此,唯一可行的策略就是基于奇偶性的策略,不存在其他策略能保证最多错一次。 附录 参考文献 由于这个问题非常经典,解法也非常经典,所以就写到这了。 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2025/9/5
articleCard.readMore

学习笔记 | 初学打字标准指法常见问题的自我 Q&A

前言 博主于 2024 年八月末开始学习打字标准指法,于 2024 年九月初提出本文中的问题,于 2025年七月中回答本文中的问题。 博主使用的是机械键盘,使用拼音输入。 我原本的输入指法十分不标准,左手四指初始位置为 Shift、A、S、D,大致是由于我是在游戏过程中熟悉电脑键盘的,而游戏的常见键位导致很容易出现这样的初始位置。并在此错误基础上,由于我后来接触了编程与算法竞赛,需要大量的输入,由此形成了一套自己的“按法”(例如输入 scanf 时我的手的动作跨度很大、幅度很不合理,但由于习惯所以打得很快,或是 G 我会下意识用右手去按)。 习惯的事情,难改是正常的,但越早改越好。要改过来也很容易,实际上打个小半天就能有点习惯了。 我是使用打字练习网站进行练习,其他的打字软件应该也是一样的效果。 个人推荐先从拼音练起,拼音是很固定的搭配组合,很容易就练得很顺手,相比之下英文与编程时可能远没有那么顺手。 据说练习时最好不看键盘 ,我持保留意见。的确你不能想着一直看着键盘打字,练熟以后肯定是不看的,但是初学时知道自己的手指在做什么也很重要,我认为偶尔以检查的目光去看一下还是有必要的。 打这篇文章时实际上还是没有完全习惯,打错很多,不过这是我没有练习的原因。 总而言之,希望能解答一些初学者的疑惑。 练习标准指法的好处 每个手指的安排相对更合理,可以减少手部移动,减轻手指负担 减少需要的位移也会间接降低输错的概率,按三格开外的键比按旁边的键要容易按错的多 熟悉以后可以加快平均输入速度 可以更充分地利用所有手指 个人认为打字时候的动作也更美观,至少你盯着自己打字的手看还蛮解压的 Q & A Q:哪只手按空格 A:左右手大拇指都停留在空格上,想用哪只用哪只。有一只说法是,按空格的上一个键是左手的话就用右手按空格,上一个键是右手的话就用左手按空格,这样会更省力,但实际上我认为怎么按都比较舒适,不会有太大的差别。 Q:无名指和小拇指感觉没力气,按键盘很虚,而且容易触发多次按下 A:这就说明以前的指法是错误的,没怎么用到这两个手指,打多了自然就有力气了,现在我的无名指和小拇指打字打起来感觉很正常,不会很虚,也不会触发多次按下。 Q:一换指法就跟不记得键盘一样 A:打少了导致的。 Q:凭什么右手小拇指在分号上 A:我是怎么问出这种问题的。右手小拇指在分号上,确实分号很少按,但它还负责了一大片区域呀,比方说回车、退格等等,实际上是最累的手指。 Q:左右手不协调 A:打少了导致的。 Q:左手中指无名指和右手中指移动的时候会连带着其他手指一起移动 A:打少了导致的。现在我不会出现这种情况,实际上出现的话也没关系,轻微动一动也可以的。 Q:总感觉某些指按某些键更合理 A:我已经记不起来那时候我是怎么想的了,应该也是打少了导致的。 Q:左手到 B、右手到 Y 太远了 A:确实,左手的 B 确实比较不舒服,右手的 Y 还好,但是习惯就好。 Q:Shift、Ctrl、Alt 哪只手按 A:如果要配合的键是右手按,那么就用左手按 Shift、Ctrl、Alt;如果要配合的键是左手按 ,那么就用右手按 Shift、Ctrl、Alt。 Q:打中文选词按数字键好远 A:这对于所有指法都是一样的吧。只能说善用空格键,然后习惯就好。 Q:老是打错 A:这不是指法的问题。 Q:打 R 的时候感觉手指有点挤 A:好像是有点,但现在已经蛮习惯的,没有觉得不舒服。所以习惯就好。 Q:手指何时该归位 A:最初的我认为打完之后手指可以在那个键上停留一会,这样如果连续打的话可以加快速度,过大概几秒手指才会归位,但是现在的我发现,打习惯之后,会感觉手指就像弹簧,你不用力的话会自动归位到初始位置,于是也就打完之后自动归位了。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2025/7/20
articleCard.readMore

逻辑解构 | 真实与谎言的守门人

题目 数学大臣尤泽尔受到国王亚路开德的召见,被侍卫克里亚带到了一个房间。在他的面前只有两扇门,门前各有一个守门人。 “两个守门人中有一个人只说真话,另一个人只说假话”,克里亚说,“陛下在其中一扇门后面等着您,但陛下要求您只可以问其中一个人一个问题。” “另一扇门后面有什么?”尤泽尔问。 这位王国最聪明的头脑,以往都是二话不说地轻松解决所有谜题,今天居然问起了失败的后果。克里亚有些诧异,但还是耐心地解释道“只有一把空椅子。” “不错”,尤泽尔呢喃了一句,令克里亚心里顿感不妙。 尤泽尔随便走向其中一个守门人,问了一个有些复杂的问题,守门人随即陷入沉思,那痛苦的表情让克里亚有些担心。但好在,几分钟过后,守门人还是回答了上来,是正确的门。然而,克里亚尚未来得及松一口气,只见尤泽尔直接打开了另外一扇门——然后坐到了那把空椅子上。在守门人赶上之前,尤泽尔砰地一声关上了门。 “让陛下来找我”,尤泽尔的声音透过门传来,“他只能问其中一个人一个问题”。 陛下应该问什么问题,才能找到正确的门? 答案(不唯一) 问其中一个守门人:“如果我问另一个守门人‘哪扇是正确的门’,他会回答哪扇门?”,回答的门就是错误的门,选择另一扇门即可。 证明 分类讨论: 若问到了说真话的守门人,则另一个守门人说的是假话,会回答错误的门,那么这个说真话的守门人就会告诉你错误的门 若问到了说假话的守门人,则另一个守门人说的是真话,会回答正确的门,那么这个说假话的守门人就会告诉你错误的门 所以无论问的是哪个守门人,回答都是错误的门,因此选择另一扇门即可。 原理 这里我们先完善一下“说假话”的定义:若答案是二元的(如是或不是、左或右等),则会说相反的答案;若答案非二元,则有可能说出任何不正确的答案。 我们先来弄清楚这个游戏存在的二元: 两个守门人,其中一个只说真话,一个只说假话 两扇门,只有一扇门是正确的 前者是游戏背景规则,也是做选择时的限制,但并非我们需要的答案;后者是游戏背景规则,也是我们需要的答案。 只问一个问题很难同时得到两个问题的答案,因此我们应该考虑放弃弄清楚谁说真话谁说假话,只要确定哪扇门是正确或错误的就可以了。换言之,我们需要问一个“不管问到说真话还是说假话的人,都会回答相同的答案”的问题。 答案的问题正是如此。从证明里我们能发现,不管问到说真话还是说假话的守门人,最后回答的都是错误的门。 为何能达成这样的效果呢?我们研究一下,问题的回答在被我们得知之前都经过了什么。以左边的门为正确的门且问到了说真话的守门人为例: 最内层的问题:哪扇是正确的门,此时回答为左 内层问题:问另一个守门人“哪扇是正确的门”的回答,由于另一个守门人说的是假话,最内层回答为左,所以此时回答为右 外层问题:问这个守门人“如果问另一个守门人…”的回答,由于这个守门人说的是真话,内存回答为右,所以此时回答为右 再以左边的门为正确的门,且问到了说假话的守门人为例; 最内层的问题:哪扇是正确的门,此时回答为左 内层问题:问另一个守门人“哪扇是正确的门”的回答,由于另一个守门人说的是真话,最内层回答为左,所以此时回答为左 外层问题:问这个守门人“如果问另一个守门人…”的回答,由于这个守门人说的是假话,内存回答为左,所以此时回答为右 可以发现,两次回答的过程的共同点是,回答都是被反转了一次的。 答案的巧妙之处在于,这个问题虽然只对一个守门人问了,但其实回答的过程把两个人都经过了,因此得到的回答一定是最内层问题的相反回答。 因此,原理同上的问题都可以完成本题的任务。如“另一个守门人会告诉我正确的门是左边的门吗?”,如果回答不会,那么正确的门就是左边的门。 思考 这里面有没有什么数学原理呢? 当然是有的。 说句题外话:逻辑解构系列里的谜题当然都是有更深的原理的,这样才值得我们研究,而且几乎必然离不开数学,这也是为什么尤泽尔是数学大臣而不是物理大臣或者解谜大臣。 理解上面的原理后,我们发现,因为原问题必定把两个守门人都经过一次,所以回答一定会被反转一次,此时两个守门人有点类似“串联”的关系。 用逻辑表示说真话与说假话使答案发生的变化的话,说真话就是保持不变,说假话就是取反。 答案问题中的“串联”,实际上就等价于最内层问题的一个取反,只不过是利用游戏规则表达成了语言而已。 同理,我们很容易想到另一种“串联”方法:将自己与自己串联。说真话串上说真话,答案不变;说假话串上说假话,就是取反再取反,也是不变。转换成语言就是“如果我问你,哪扇是正确的门,你会回答哪扇门?” 这样我们就得出了本题的另一个可能的答案,并且这个答案会直接得到正确的门,不用再反转一次,并且哪怕只有一个守门人也可以问到正确的门。 和位运算有关系吗? 过程之中的反转,很容易让我们联想到布尔值。提到布尔值,很容易联想到的就是逻辑运算或者位运算,这里我们研究位运算,那不得不做的就是看一下真值表。 以二元问题为例,我们把其中一个答案看作 $1$,相反答案看作 $0$。这里我们将一个问题在经过守门人之前的答案看作输入,守门人会给出的答案看作输出。 那么,只说真话的守门人的真值表: 输入 输出 0 0 1 1 原封不动,我们可以看作和 $1$ 做与运算,也就是 x & 1。 我们再来看看只说假话的守门人的真值表: 输入 输出 0 1 1 0 取反,我们可以看作取反,也就是 ~x,也可以看作 x ^ 1。 由此,我们可以发现上面的第一种答案对应的运算其实就是 ~(x & 1) 或者 ~(x ^ 1),第二种对应的是 ~(~x) 或者 x ^ 1 ^ 1。 由此我们也可以玩的更复杂一些,叠更多的运算。当然,由于我们并不知道哪个人说真话哪个人说假话,所以两种运算符的数量要一致。 例如,我们尝试构造 ~(~~x&1&1)&1 对应的问题: 问说假话的人:如果我问另一个人【如果我问另一个人〖如果我问你“如果我问另一个人‘如果我问你,哪扇是正确的门,你会回答哪扇门’,他会回答哪扇门”,你会回答哪扇门〗他会回答哪扇门】,他会回答哪扇门? 更多的守门人? 位运算总共有六种,运算对象有两种,由此可以构造出一共八种守门人,请选择你的角色: 位运算 对应的守门人 x & 1 只说真话 x & 0 只说否 x | 1 只说是 x | 0 只说真话 x ^ 1 只说假话 x ^ 0 只说真话 ~x 只说假话 左右移 胡言乱语 真的没办法同时问出谁说真话谁说假话吗? 倘若限定只问二元问题,那确实是没办法的,因为规则限定只能问到一个二元信息,只能表示一个回答。 若不限定只问二元问题,那就需要对“说假话”有更进一步的定义了,在特定定义下像这样的问题可能会有效: 若你说真话且这是正确的门,请说 1;若你说真话且这是错误的门,请说 2;若你说假话且这是正确的门,请不要说 3;若你说假话且这是错误的门,请不要说 4。 但我更倾向于,“说假话”的定义应该可以解释为:先假设你说真话,得到的答案我们定义为正确答案,则你可以回答除了正确答案以外的任意答案。 在这个定义下,上述问题也无效。 若答案只有两种,则只能表示一个回答;若答案有两种以上,那么说假话的人,除了正确答案可能会回答任意一种,无法和情况一一对应。也就是说,以最严格的规则是无法同时问出的。 附录 参考文献 问题经过故事化(这会成为逻辑解构的传统),原问题与原答案来自于某本小说。 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2025/7/4
articleCard.readMore

逻辑解构 | 默契硬币

题目 国王亚路开德试图量化一对搭档有多默契,于是他设计了一个默契测试:两位受试者分别在两个相隔遥远的封闭房间内各抛一枚硬币,然后猜测其搭档的抛硬币的结果,如果两个人中至少一个人猜对搭档的结果,则计一分。进行一百次这样的测试后的总分,就是这对搭档的默契值。 然而,数学大臣尤泽尔立刻发现了这个测试的漏洞,他想到了一个可以百分百得分的策略。 请问尤泽尔的策略是什么? 答案 一人始终把自己硬币的结果猜上去,一人始终把自己硬币的相反结果猜上去。 在这样的策略下,总会有一人猜对,一人猜错。 证明 设 A 是那个始终把自己硬币的结果猜上去的人,B 是始终把自己硬币的相反结果猜上去的人。 不妨设 A 抛到了正面。根据 A 的策略,A 会猜 B 抛到了正面,此时有两种可能: 如果 A 猜对了,即 B 抛到的是正面,根据 B 的策略,B 会猜 A 抛到了反面,B 猜错。 如果 A 猜错了,即 B 抛到的是反面,根据 B 的策略,B 会猜 A 抛到了正面,B 猜对。 所以总会有一人猜对,一人猜对。根据游戏规则,在这样的策略下可以百分百得分。 原理 先来看最容易想到的一种错误方式:每个人猜对的概率是 $\frac12$,那么两个人中至少一个人猜对的概率应该是 $\frac34$。如果这是对的,那么不应该存在百分百得分的策略才对。 答案的正确性毫无疑问,但其并非来自于灵光一现,下面我们来理解答案的原理。 答案是怎么起作用的?上面的想法错在哪儿了?为什么猜对方的硬币,策略却只与自己的结果有关? 硬币只有两种结果,所以一共只有 16 种情况,我们可以使用枚举法。这里我列出了所有无法得分的情况,即两个人都猜错的情况: 序号 A 结果 A 猜测 B 结果 B 结果 B 猜测 A 结果 1 正 正 反 反 2 反 反 正 正 3 正 反 正 反 4 反 正 反 正 如果你观察力足够强,可以发现四种情况存在一些共性: 1 和 2 是两个人都把自己硬币的结果猜上去了 3 和 4 是两个人都把自己硬币的相反结果猜上去了 现在可以回答第一个问题了:答案的策略完全避开了两个人都猜错的情况,所以就必然得分了。 的确,枚举之后就可以瞪出答案。直接枚举情况然后针对特点来设计策略是一种好方法,但可能会有点令人失望。有没有更深一点的原理呢? 还可以这样理解答案:”把自己硬币的结果猜上去“有些奇怪,其实意思就是“猜两个硬币的结果相同“。同理,另一方实际上在做的是”猜两个硬币的结果不相同”。显然,两个硬币的结果要么相同,要么不相同,所以必然一人对一人错,因而必然得分。这么一讲就非常好理解了。 要回答第二个问题,我们需要用新的理解去重新看待这些事件。 抛两次硬币,结果相同的概率的确是 $\frac12$,也就是这种策略下 A 和 B 猜对的概率确实都是 $\frac12$,但不同的是,这时 A 猜对和 B 猜对是一对互斥事件,而非我们先前认为的相互独立。 独立与互斥,显然是策略不同导致的。随机猜时确实相互独立,但为什么这个游戏里可以按某种策略然后变成互斥? 关键在于,正如上表,这个双人游戏实际上并非简单的猜两次抛硬币,而是包含了两个信息与两个决策,总共四个事件的。 如果受试者随机等概率地猜测,信息与决策无关,四个事件之间相互独立,所以十六种情况概率相等,于是得分的概率就是 $\frac{12}{16}$。思维盲点在于,四个事件在游戏中并不完全独立,游戏中存在一个很容易忽视、极其隐秘的联系:猜之前是已知自己硬币的结果的。 画成流程图则如下: 有已知信息,那么决策时就可以考虑利用已知信息,从 16 种情况中划去一些情况。 例如,若 A 的策略是只把自己硬币的结果猜上去,那么 16 种情况就会划去 8 种,剩下 8 种情况。而上表 4 种不能得分的情况也被划掉了两种,只剩下上表中序号 1 和 2 的情况。 不难想到,假如 B 也有一个利用已知信息的策略能够把 1 和 2 给划掉,同时可以得分的情况没有被划完,那么就可以实现百分百得分了。 观察序号 1 和 2 中 B 的结果与已知的关系发现,B 都是把自己硬币的结果猜上去了,不难想到 B 的策略:只把自己硬币的相反结果猜上去。不能得分的情况全被划掉,同时还剩下 4 种可以得分的情况,这样就必然得分了,也就是我们的答案。 第三个问题也迎刃而解:因为在猜之前,自己的结果是一个已知信息,既然要有策略,那么当然就只能是由已知信息来决定猜什么。 好像划掉情况很容易呀,那还有没有别的答案呢? 先来个简单的:A 只猜正,也就是把自己身份也当做一个已知信息来决策。我们把前面不能得分的四种情况再列一下: 序号 A 结果 A 猜测 B 结果 B 结果 B 猜测 A 结果 1 正 正 反 反 2 反 反 正 正 3 正 反 正 反 4 反 正 反 正 A 只猜正时,总情况数少一半,上表中四种情况只剩下 1 和 4。那么此时 B 能否找到对应的策略呢?答案是不能,因为情况 1 和 4 中,B 自己硬币结果都是反,但 B 猜测结果有正有反,两者找不到关联,没办法形成从已知推决策的有效策略,他只能知道自己抛到正面就稳了,但是抛到反面的话,B 没办法依据已知信息来决策自己猜正还是猜反,不能划掉这两种情况,因此不能必胜。A 只猜反也同理。 从这里我们可以发现,上表四种情况我们要么剩 1 和 2,要么剩 3 和 4,否则会出现 B 自己的结果与 B 的猜测结果找不到关联,那么 B 决策时就利用不了已知信息,那就变回了独立的情况了。 因此,没有别的答案,只能是答案的策略或其等价形式。 思考 这还是抛硬币吗? 答案的策略,实际上将问题转化为了猜两个人抛硬币结果相不相同,为什么可以这样转化呢?两者之间有什么隐含的联系? 我们来看看这两个游戏分别都发生了什么。 采取答案的策略后,只剩下四种可能情况,我们一一列出: 序号 A 结果 A 猜测 B 结果 B 结果 B 猜测 A 结果 结果 1 正 正 正 反 A 猜对 2 正 正 反 正 B 猜对 3 反 反 正 反 B 猜对 4 反 反 反 正 A 猜对 因为 A 与 B 的猜测来自于 A 与 B 的结果,是确定的,因此我们不妨略去这两列,即: 序号 A 结果 B 结果 结果 1 正 正 A 猜对 2 正 反 B 猜对 3 反 正 B 猜对 4 反 反 A 猜对 抛两次硬币也是四种可能情况,我们列出: 序号 第一个硬币 第二个硬币 结果 1 正 正 相同 2 正 反 不同 3 反 正 不同 4 反 反 相同 很容易发现,之所以能这么转化,是因为答案的策略划掉了十二种情况,剩下的四种情况与抛两次硬币的情况相同,因而我们可以这么转化地去解释,但两个问题并没有本质上的关联。 那么,我们能否设计出类似的游戏呢? 实际上只要是类似于这个流程图的,都是一样的策略。 例如,两个相隔遥远的人,互猜对方的性别。 有没有什么数学原理呢? 我们不妨转换为布尔值(硬币正面为 0,反面为 1),列出 A 与 B 策略的真值表。 A 的策略: 已知 猜测 0 0 1 1 B 的策略: 已知 猜测 0 1 1 0 直接说 A 是相同 B 是相反好像并没有发现什么新的原理,因为策略本来就是这么描述的。所以我们往其他方向考虑:奇偶性? A 的已知与猜测的和是 0 或 2,都是偶数;而 B 的已知与猜测的和是 1,是奇数。因此我们可以这么描述策略:A 始终猜两个硬币的和是偶数,B 始终猜两个硬币的和是奇数。由于两个硬币的和要么是偶数要么是奇数,自然总有一个人对。 上面的思路看起来很有空间,因为奇偶性其实就是进行模二,而模 $n$ 运算具有天然的 $n$ 分性。 改成模三是不是可以拓展到三个呢? 当然是可以的。 流程图表示出来较为复杂,设计起来也是。这里有 $3^6$ 种情况,采用决策过后有 $81$ 种情况,只需让每个人占有其中 $27$ 种情况作为胜利即可。 我们依据这个图设计一个最简单的、没有玩什么文字游戏、没有隐藏信息、没有加太多限制的游戏: 三个人各戴一顶红色或黄色或蓝色的帽子(颜色可以重复),每个人只看得到别人帽子的颜色,看不到自己的,不能进行任何形式的交流,若三个人中有人猜对自己帽子的颜色即胜利。 类比上述数学原理,我们设红为 0,黄为 1,蓝为 2,设三个帽子的和为 $s$。 由于 $s\bmod 3$ 的必然是 0 或 1 或 2,我们容易想到以下策略: A 始终猜 $s\bmod 3\equiv 0$ B 始终猜 $s \bmod 3 \equiv 1$ C 始终猜 $s \bmod 3 \equiv 2$ 这样就必然有一个人是猜对的。同理,可以推广至更多人的情况。 当然这个游戏是我现场设计的,不要纠结里面的漏洞。 套娃:数学原理的数学原理 如果你深究上面的策略究竟是不是对的,一定会注意到策略成立的一个前提:ABC 这样的“猜”等价于某一种符合游戏规则的策略,即,这样的策略在游戏中能对应唯一确定的颜色。 我们可以直接列。由于另外两个帽子的和模三一定是 $0,1,2$ 之中一个,且自己的帽子也只能是 $0,1,2$ 之中一个,所以: 若 $s \bmod 3\equiv 0$,则 $s$ 来自于 $0+0$ 或 $1+2$ 或 $2+1$ 若 $s \bmod 3\equiv 1$,则 $s$ 来自于 $0+1$ 或 $1+0$ 或 $2+2$ 若 $s \bmod 3\equiv 2$,则 $s$ 来自于 $0+2$ 或 $1+1$ 或 $2+0$ 例如,看到的是一个红帽子和一个黄帽子(总和为 $1$),并且我猜 $s\bmod3\equiv0$,那么可以算出自己是 $2$,即蓝帽子。 实际上,这就是一个模运算的填空题:$1+\square\equiv0\pmod 3$。 在 $\square\in{0,1,2}$ 的情况下,类似这样的填空题中 $\square$ 是有唯一解的。也就是说,在已知 $s$ 并已知另外两个帽子的和时,确实可以唯一确定自己的颜色出来,该策略有效。 附录 参考文献 本题经过了我的美化(当然,这会成为逻辑解构系列的传统),第一道思考题来自 GM 的视频。 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2025/6/30
articleCard.readMore

拾光纪行 | USTC 强基计划游记

前言 距离高考大约一个多月的时候,开始报名强基计划。 综合各学校、各专业以及我的倾向,最终选定了 USTC。我对 USTC 最初的好感来自于很帅的名字,在我眼里仅次于 THU;其次是在人工智能我喜欢的方向上 USTC 有先进的研究;最后是听说 USTC 的学生浓度都…比较有生活( 虽然对我来说难度很高,能接受的专业也仅有一个,但抱着“文化考得上的大学不值得我读十年”以及出于中大的保底情况,所以还是选择了这一所我好感度很高、可以说比较向往的学校,尽管是冲全广东只招一个人的专业且不接受调剂。 尽管合肥目前给我的印象其实不是很好,相比深圳确实没有那么舒适和现代化,但还是选择相信 USTC。 实际上现在并不知道未来会如何,落榜的可能性远大于录取,并且在一个相对没那么喜欢的城市待十年对我来说也很难直接接受。USTC 的强基会锁档,出分前完成所有流程,然而我只要在笔试、面试、高考之中任意一个爆炸就会直接离场。 我尽力去尝试,不管结果如何,我相信以我自己的能力,大学过后都能走到我应该的结局。 笔试 2025 年的笔试是在 6 月 15 日,我 14 日上午从深圳宝安机场出发飞抵合肥。 下午去东校区踩点,不知道东校区是不是本校区,但是这个大门感觉还是比较…低调。但附近配套感觉除了“科气”十足以外并不称得上舒适,可能比起出校门,选择点外卖会更科学一点。 踩完点之后去附近的南七天地逛了一下,以为是很大的超商,可能是我去的时间不对,去的时候人很少而且很多很多店铺都没开,感觉还是有些失望。什么吃的也没买到就出去了,在旁边另一个商场逛了一下发现不卖吃的,只好打道回府。顺带一提,那里的地下通道是真的有些…不合格。 最后发现反而走到酒店附近才找到吸引我进去吃的店,后来我才注意到原来我酒店那条道叫做美食街。 晚上继续复习,复习的主要是物理,因为以为会考竞赛知识点,所以这三天临时补了很多。然后看了一下物理数学近两年的真题找找感觉,就结束复习了。可能是因为我已经考过一次 SUSTech 的机试(其实被我当 USTC 笔试的模拟赛打了) 第二天一早就起来,然后提前了一些出发,也提前了一些到考场。什么,人怎么这么多。 进考场之后,监考老师提醒我们 17 号查成绩,21 号面试,注意消息。等了半个小时后就开考了,大致流程应该在尽量模仿高考,但不时露馅。 先考 100 分钟数学,提前七八分钟拿到卷子开始浏览题目,有十道填空和两道大题。我从第一道题开始看,看了一会有思路了就往下过,一直过到了第七八题才打开考铃,也就是说还没打铃我就已经会做七八道题了,芜湖起飞。 考前想的思路大部分都是对的,最终都得到了答案,虽然有些题我不记得我的答案是多少,后续对答案也对不了,但我觉得应该是没问题。不会的题有两道,一道是球盒的期望问题,可以写出式子但化简时无从下手,最后蒙了答案但是错了;另一道是奇怪数列吧好像,有 sigma 和一个一九九几年的数字的题,一开始的思路错了,后面发现不会做于是等到考试快结束的时候感觉回来列了前三个找规律,但因为时间不够,第三个没算出来,又忘了第一个,所以找了个很错的规律,打铃之后才发现存在规律,当时很遗憾,后面发现那个规律也只有前三个是对的,释怀了,不会就是不会。 值得一提的是我很喜欢斐波那契数列那道题,虽然被学 OI 出身的我读完题后就感觉到了很浓的递推味然后秒了,但那道题的设题方式确实很新颖很有创意。 大题的话,第一道大题是复数大题,最后应该会变成数论,我数论都是从 OI 学的,做不来,跳。第二道大题发现讨论就可以了,咔咔写,但是脑子有点点转不动所以写的比较慢,挺久才写完,在这里应该浪费了不少时间。 10 分钟休息过后又是 100 分钟的物理,我不想评价,没考太多竞赛但一点也不简单,而且还有多选题,可恶。我感觉我最有把握的题是多选最后一道,虽然那道题题面有误,能量标错了,但是不影响答案。其他的题基本连蒙带猜,大题也根本写不动,把能理解的写了就结束了。 后来听说物理好像不答不加也不扣分,但答错还要倒扣一分,我当时没注意看。我只一直盯着多选题里面那个“可能有多个答案“的”可能“然后在想他会不会在多选题里出单选题。 考完之后有一种解放感,去找女朋友玩了好久。17 号从早上开始时不时刷新,最终是晚上在女朋友家刚吃完饭已刷新看到我入围面试,蛮开心的。 数学 70 分,满分 100 的这个难度能考到这个分数我还是相当满意了,也就是说我写的大部分都是对的。物理 34,意料之中。笔试总分 104,入围线 70,那我岂不是不用考物理也能入围。 毕竟我学过 MO 和 IO 但没学过 PhiO,所以我觉得这样的分数还是意料之中的。不知道这个入围线能不能参考,但按照往年的分数线,其实感觉 104 分竞争力不是很大,也同样是淘汰边缘了感觉。 不过我听说我的竞赛奖在面试环节可以加分,希望如此。好好准备面试,等到出分后就知道结局啦! 话说看知乎,之前强基有礼物,今年怎么没有?是要等到面试才有嘛,期待一波。 面试 没有礼物,可恶,为什么今年没有,为什么为什么为什么为什么。 21 号上午面试,我 20 号上午面试完 SUSTech 直接前往机场飞到合肥。 上午应该都是第一类考生,大概有两个考场,每个考场约 108 个人,分为三轮,每轮三组,每组 12 个人。 大概面试流程:12 个人分为甲乙两组,先五分钟阅读和讨论材料 A 及问题,然后 1-6 号轮流发言,接着 12-7 号轮流对前面同学的发言提出问题,发言都是两分钟,然后再给五分钟阅读和讨论材料 B 及问题,然后 7-12 号轮流发言,6-1 号轮流提出问题,也都是两分钟。 材料 A 与 B 印在同一张纸的正反两面,另给了两张白纸可供记录,需自己带笔。不能提及自己身份和高中,所以当然不需要自我介绍。 今年我的面试真题两则都是科大今年的创新成果,材料 A 是可用于抓握复杂或微小物体的对数螺旋软体触手,材料 B 是利用谐波衰减速度不同以及拍频效应来减少纳米波雷达的心跳谐波监测受到呼吸谐波的干扰,问题大致都是些有关专业、进行联想、推理原因、探索应用之类的。 个人认为提前押材料并没有意义,因为考场给出的材料基本都有具体的点,还是要依据具体问题和材料原文去回答,像我这次材料 A 早在很久之前在 b 站发布视频的时候我就有看过,但个人感觉除了稳定信心以外没有实际帮助。 值得注意的是,如果你是提出问题的一方,有必要认真先前发言的同学的内容(包括对方和己方),这样你才有针对性的回答,或者在合适的时候提到哪个同学发言的哪个内容,告诉老师你认真听别人发言了;如果你是回答问题的一方,一定要在讨论阶段和大家分配好每个人的角色,比如四个问题六个人怎么分配,可能一个问题里有两个小问要不要拆开来回答,或者需不需要一个人负责补充或总结,最好是在准备阶段简单记录下自己的发言要点,然后也可以记录一下前面同学的发言,听发言的时候也可以思考一下别人的问题,在自己发言的时候可以提出补充。 个人觉得,发言的时候客套话(老师好同学好谢谢之类的)适量就好,但是重要的是发言流畅,有逻辑性。可以说慢一点,但是嘴巴最好不要停,脑子暂时跟不上的时候就随便塞一些有逻辑的空话进去(考虑到技术可行性、考虑到实际情况、根据前面同学发言、根据材料之类的),另外说话应该谦虚友善一些,毕竟这不是辩论只是小组讨论,应该使用“我认同对方同学的观点,但我觉得还可以考虑”、”对方同学的角度很好,我可以补充“、”我认为/我觉得/我的看法是“、”对方同学的考虑有一定合理性,但我有别的看法“、”对方同学的看法可能…,我们不妨…“之类的语言,要避免使用强烈的否定甚至攻击对方。一定要知道不是在辩论,只是小组讨论,让你提出问题不是和对方辩论,对方发言没问题或者很好的部分应该要认同对方观点甚至可以帮对方补充。 理论上非金银牌的竞赛获奖的加分会在面试分数中有所体现,但这只是理论。面试不会提前出分,会在高考出分后直接与综合成绩一并公布,并在高考出分几天后公布各省各专业录取线以及可以录取情况查询方法。有一说一,强基招那么少人,应该基本每个省都是每个专业只招一个人,这个公布录取线基本就是公布录取的人的综合分了吧。 这个综合分其实很不好估,往年录取分数线参考意义也不大,因为综合分的三个部分中面试成绩你不知道,而笔试和高考每年难度都不同,尽管相对排名可能不变,但在综合分中的参与得分情况就会不同,所以具体分数线无法比较。 个人感觉发挥的还不错。笔试+面试的两分就顶高考一分了,希望我的省一可以给面试多加点分,这样可以补一补笔试和高考的漏分,别白学三年竞赛。 虽然其实录取概率极小,但不管结果怎么样我都已经尽力了,rp++ 吧。 出分 高考分实在有些炸裂,无缘 USTC 了。 结果 未录取,但是也不算太遗憾吧。除了这次没有礼物。 面试分 55.5。 综合分相差的有点多,约等于三十分高考,或是六十分校测。 就算我发挥好了也考不到这么高,这么看来爆炸的其实不只有高考,基本是全炸,所以也就不遗憾了。虽然是三年以来很喜欢的学校,但倒也称不上梦想吧,无非就是好感比 SYSU 高一点所以挣扎了一下罢了。 过段时间还会写 SYSU 综评游记。 顺带一提,填这个志愿是真的烦。 附录 依稀记得报考 USTC 羟基前看了一篇往年学长的知乎文章,与我的结局十分类似。 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2025/6/18
articleCard.readMore

随笔杂谈 | 2025 新春快乐

依旧是不知不觉间,博客已经过去了千天的四分之三。在去年里博客总共更新了十七篇文章,收获了许许多多的互动,还是相当满意的。 2025 年,予我而言是人生迄今为止最为重要的一年,我也正在努力着,期待今年六月的结果如何。 每个人的一年里真的会发生很多很多事情,时间滋润着成长,不管是我还是这篇博客,亦或者我在元旦之日种下的六株向日葵。 但无论如何,祝各位新的一年巳巳如意、生活美满;多喜乐,长安宁。 健康、幸福、进步。

2025/2/8
articleCard.readMore

拾光纪行 | NOIP 2024 游记

游记 作为一个高三半 AFO 选手,这次 NOIP 意味着彻底的 AFO 和全面的 whk。 赛前 day -3 中午开始停课。 教练安排我和另外一位同学在明后两天组两场模拟赛,我的题倒是提前组好了,不过感觉 A 题放了绿题会难过头,所以临时换了一题黄题。(然而 NOIP A 题蓝,哈哈) 下午帮同学证明代码正确性,然后在讨论区看到野题拉同学想了两小时,结果那个帖子被删掉了。后面在讨论区看到了你画我猜,第一次见到信息竞赛词库,挺好奇的,进去看了一下,结果一不小心玩了快两个小时,哈哈,太好玩了。 顺便也靠这个重新回忆起了许多算法,原来我三年来已经学了或者了解了这么多,十分感慨。 晚上在机房组了一场模板手速赛,基本十分顺利的写出来了。除了树状数组不小心把 $m$ 打成 $n$ 以及第一次写 $O(n)$ 建树,还有字典树的映射写挂,其他模板都一次过了。不过手速不够所以一些图论高级算法的模板没打到。 day -2 上午验了模拟赛的 C,中午开始考。然后经典赛时开始造大样例,以及赛时才验 D(本来几乎一次过的,结果因为不小心把 min 写成 max 调了好久)。 晚上给机房讲题,讲到喉咙冒烟。组比赛真的比打比赛累多了。 day -1 上午有点摸鱼,下午打同学组的模拟赛。A 因为差点没开 long long 差点见祖宗,B 是我推荐给他的题所以原了,C 是缩点板子所以二次原,D 是万恶的小木棍。 通过洛谷帖子保存站,发现前天那个帖子删之前有人回答了,并且做法大致和我想得一样。然后找到了那个你画我猜的帖子,但房间已经没了所以自己创建了一个。 和机房同学试了几个词库之后回到了 OI 词库,然后人一下子变多了起来,好玩。后面有点累了,想着等再轮到我画一次就退。结果中间有个人画了矩阵快速幂,所有人都答出来了只剩我,然后我猜了各种“矩阵优化dp”“矩阵加速递推”直到有人提示我“矩阵什么也没优化”才压着时间线答出了正确答案,他们甚至发起投票踢我,然而我是房主,乐。(伏笔) 晚上对着题解把小木棍调出来了,并且成功快速理解了那两个优化。后续接着打图论模板,除了欧拉路径有点模糊以外都很顺利的打出来了。 放学前半小时在机房里扬言要挑战 15min 敲树剖板子,结果因为太久没打所以手疏导致实际用了一倍的时常。打完之后成功过了样例,提交发现最后一个点没过。可恶,差点一次性过掉了树剖。 后面看讨论区才知道最后一个点是 hack 没对输入取模的,长记性了。 day 0 顺利起床,吃完早餐后坐校车前往考点。 在考场门外面基了某位神犇数秒,大致流程是:突然被手机贴脸->递东西->消失。 进考场,发现挡板不太行,能够清楚看到斜对角两位选手的屏幕,但看得出来主办方已经尽力遮挡了。 我们学校和另一强校交替着坐,感觉压力很大。而且复赛时候的咬笔哥这次来到了我的左侧,并且我依旧不知道他是否有第二支笔。 签字,话说今年居然不用填“我选择在 Windows 系统”,以及经典提前三分钟发密码提前三分钟收卷,到底有什么必要嘞,还导致我们开始前三分钟不能建文件夹。 看到这解压密码,感觉有点东西。莫名有点应景(指 AFO 前最后一战) 赛时 开 A,感觉有点难办。来来回回只能往贪心上想,去年的这个时候已经写完 A 了。 记得最开始写了一版贪心,但是后面发现假了,但是改一改就真了。一个半小时才敲完了,是现在主流的统计区间个数然后双指针的做法,我以为很对,结果大样例第 7 个和最后一个过不去,莫名其妙多了一点。 对着代码找不出错,想过写对拍但发现我不会暴力做法,于是只好硬着头皮去看大样例,但好在 7 是特殊性质,比较简单。 分析了 7 后手动模拟发现手算出来的就是那个错误的答案,很绝望,这可能说明我贪心假了,此时已经两个小时了,去年这时候我马上就要调出来 B 了。 想着至少能过好多个点,先去看 B 了。读完 B 感觉,完了,B 也不会,甚至不知道读没读懂。一开始以为是推出式子快速幂算就好,仔细读题之后分析了一下发现不是简单的相乘,要分情况递推。 因为感觉最后肯定是想办法压缩不确定的一段的方案计算,于是就先对 $n$ 写了个 dp。 可能是由于昨天你画我猜里矩阵快速幂给我印象很深刻,写完之后定睛一看,发现这状态转移式子长得很递推啊我啪的一下就改成了矩阵。卧槽那这不快速幂秒了吗,喜滋滋写完之后稍微一挑就过掉了大样例,芜湖。 还剩一个半小时,打算把后面两题暴力和特殊性质拼了再回去调 A。 C 题想了一会写出来一个式子,感觉是换根 dp 但我不太会,所以想着写个树形 dp 的暴力,结果写了四十分钟发现假了,只写了个链,然后菊花图的式子也假了,绷。 D 题很优雅,但是不会,看特殊性质火速打了链,但是是 $O(qn^2)$ 的暴力取 min 和 max,哈哈,炸裂。而且大样例不知道为什么会卡死,运行不了。 最后二十分钟调 A,然后这时候我终于想出来了错在没有考虑相交区间长度的限制,但是我以为这里要做决策来决定怎么分配 0 和 1 的比例,所以以为贪心假了。但是十来分钟也改不了什么了,所以写了个数学里的分层抽样一样按比例的神秘决策,结果过掉了大样例。 最后十分钟检查代码的,找到了 D 应该是导致卡死的原因,改了编译了但没测样例。最后三分钟想把 A 的 memset 改成 for 来初始化,最后勉强卡着时间改完了,能过编但是也没测。 出考场之后才回想起来 $T\le 10$,memset 根本不慢,后悔改了。然后这时候悟到了提前收卷的意义,因为这样哪怕结束后才保存的话也不会判时间无效。 赛后 出考场后 考完后听到很多人说很高分,压力。在考场外地板上发现了野生零食,乐。车上吃了好多零食,得知机房其他人考的也很炸裂,悲。 晚上过掉了 B 题的自测,舒服了。 次日 后面证出来了 A 的那个决策不重要,不影响答案,怎么选都行,于是乎就连最神秘的我的按比例选也能过掉,哈哈。 中午过掉了 A 的自测,舒服了。C 的链原来只有 4 分,好吧,合理。D 的暴力仔细一想好像其实估计没有分,悲。 退役很久之后 最终成绩 $100+100+4+0=204\text{pts}$。 实际上在上面的内容是去年十二月初写在洛谷的,但时隔很久(次年年初)才发布。不过并不是因为不接受成绩或者 AFO 伤感之类的,只是单纯没有机会转移到博客上来。 “没有机会”指的不是没有时间,尽管我确实更多忙于文化课。也不是懒,好吧感觉确实更贴近懒。 学习 OI 三年,不长不短,按理说 AFO 之后应该需要写一篇 OI 生涯总结,也许不久就会开始写吧,虽然已经 AFO 有点久了。甚至现在的时间点,比起 AFO,可能离我高考后回归 OI 的日子还要更近。 虽然 AFO 且时隔久远,还是多多少少复盘一下。首先这次最大的失利就是我在第一题上花的时间不太合理,并且我认为其中一个细节其实是很难想清楚的,导致我考场上不能肯定我的贪心是错的。我总觉得不太签到,但毕竟竞赛不是小孩玩的游戏,有选拔的目的在的。然后第二题能够写出来,真的很有运气的因素(详见 day -1 的伏笔),否则我真的没有信心能够在考场上想出正确的做法。第三第四题一如既往地不善于写部分分导致错失一等,不过大学比赛就不需要写部分分了,哈哈。 虽然认为没有以一等作收尾有些遗憾,但是也完全能接受。三年 OI 最终至少给我带来了相当于保送中大计算机系的机会,所以也算是减少了一些心理上的压力。尽管如此,我也还想尝试一下靠文化课能否达到更高的高度,所以仍然在加油的路上。 尚有星光伴前路,心向远方更前行,以此作结。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/12/1
articleCard.readMore

拾光纪行 | CSP-S 2024 复赛游记

day -1 退役前的最后一场 CSP-S。听说 kkk 今年可能不会提前公开选手代码,坏。 白天忙着组了一场模拟赛,晚上自己复习知识点。 重新理解了一下三种联通分量,然后熬夜到了两点。 day 0 上午 组模板手速赛但是只有我打,大部分都打的很顺利。 LCA 写挂了,发现是忘记初始化 $dep_s=1$。之前也在这里挂过,当时没理解原因,这次好好想了一下发现是因为 $dep_0=0$,如果 $s$ 深度不设成 $1$ 的话就和 $0$ 点区分不开,跳的时候会跳错到 $0$。 线段树 2 写挂了,以为是懒标记的问题,调了半天,最后发现是把 add() 复制出来改成 mult() 时忘记修改函数内递归的函数名了,我说怎么退化成区间加了。其他部分都没问题,很好。 KMP 几乎不记得了,坏。重新看代码理解了一下,发现很好理解。 中午 出发去考场。路上重新复习了一下三种联通分量和对应的 Tarjan 算法,看了一眼 Nim 游戏结论。最后就在复习数位 dp,然而直到进入考场都没完全理解。 由于准考证和行李之类的问题在考点大门和考场来回跑。 今年还是不让带零食。还有五分钟时进了考场。 位置还是在角落,斜对角就是一位特别强的学弟。 赛时 提前三分钟发卷,压缩包和 pdf 密码都一次性打对了,rp++。 敲了代码缺省,开 T1。前五分钟想复杂了,思路有些乱,想上数据结构。重新读题发现就这,很快就会了排序贪心的做法,决定先过掉 A 再开后面题。开写的时候注意到值域可以直接桶排,14:41 的时候过了所有样例。 开 T2,怎么是物理?发现超速区间很好推,显然转化成区间覆盖问题。第一问可以离线下来用前缀和来求区间内是否有摄像头,线性。第二问读完就想到了种树,给我一种按端点排序贪心很对的感觉,但是不太敢写。 想了二十分钟,第二问还是不知道怎么做,遂跳至 T3。读完题很有思路,马上就想到了 $dp_i$ 可以分成 $dp_{lst_i}$ 和 $[lst_i+1,i-1]$,后者用前缀和很可以维护,但唯独 $lst_i+1$ 这个点的贡献到处飞,很难搞。 想了一会不会搞这个点,但是觉得只要搞出来就可以 dp,开 T4 一眼不会,遂回到 T2。 虽然一开始贪心思路记反了,按左端点排序但是取了最右,过掉了小样例,遂开写,发现过不了第二个样例。 对样例调了一下发现贪心错了,想了一会想不到别的贪心,但发现如果按右端点排序两个样例都能过,遂改。改完之后过掉了所有样例,但发现 T 了。 于是优化掉了 vector,然后其他地方卡常了一下,同时确认了清空和边界之类的细节,大样例 1.7s 过了。此时 16:43,也就是花了两个小时。 回到 T3,然后思考了很久 $lst_i+1$ 这个点要怎么搞,甚至想过退化成记搜或者上个数据结构,想了大概半小时后灵光一闪发现了 $dp_i$ 和 $dp_{lst_i+1}$ 的结构是一样的,给状态加个不影响的条件就可以直接从 $dp_{lst_i+1}$ 转移了。 这样的话就很好写了,码码码,发现第二个样例没过,对着调了一会发现对 $dp_{i-1}$ 取 max 就能过,虽然当时不理解要如何解释,不过写上之后过掉了所有样例,此时 17:40,剩余不到 1h。 后面就很坐牢了,完全把题读懂花了一些时间,甚至不会朴素暴力,只会一个 $O(n^2logn)$ 的树上大模拟,加点时从叶子向上 vector 合并。感觉应该有很多分,遂开写。由于建树的时候写了类似动态开点线段树的写法,所以被如何计算是第几场比赛难住了好久。 最后五分钟的时候放弃了,无奈 puts("1")。最后两分钟检查了所有文操、注释、函数返回值(高一省选的时候就因为把 void 打成了 int 又没有返回值导致那题 CE 100->0)。 最后开了 checker 确认了一遍文件名和准考证号,铃响结束。 赛后 本次比赛遇到三位莫名其妙像熟人一样跟我搭话的,以及对面的大哥全程嘴里叼着笔,也许他不止一只,以及右边的大哥总感觉经常瞅我进度,也许被我压力到了( 过程中经常观察学弟的表情,看起来很轻松,以为被单调队列了。 出去后吃了点零食重启一下脑子,回去路上晕车但是刷洛谷讨论区。貌似机房其他人没有会 C 的,希望他们也能 1= 吧,rp++。 发现全机房只有我一个人考完就返校,其他人都回去睡觉了,愤愤。 先回机房吃了点面包和酸奶垫垫肚子,发现自测快开了,然后开始默写代码和写题解。默写的过程中发现我 T2 虽然已经很小心但还是有地方没处理好边界,最后效果相当于默认 $0$ 处有摄像头,一旦有车在 $0$ 位置超速而 $0$ 位置没有摄像头,答案就错了,寄。 当晚自测结果前三题都没挂,甚至第四题出民间数据之前在洛谷上还排到了 rk3,激动。 后日谈 后续在多个平台自测均为 $300\text{pts}$,第二题也过掉除了我以外的所有 hack 数据。也许是因为卡掉我的情况算非常 corner case,所以非刻意构造的话很难出现,希望 CCF 也不要卡我。 中午敲了 $400$ 下电子木鱼,结果 CCF 连续诈骗两次,然后下午一赶到机房就听到了同学的祝贺。 $100+100+100+0=300\text{pts}$,达成三年 CSP-S 都没有挂分的成就。 虽然省排有些烂,但分数还不错,第四题还是黑题,还是很满意了。 希望 NOIP 可以继续发挥好,rp++。 update:今年 GD 高中生分数线居然要 $100\text{pts}$,并且刚好 $100\text{pts}$ 的还要求初赛 $\ge 61\text{pts}$,放以前想都不敢想。虽然我的初复赛都超线很多,但是机房同学好多卡线然后因为初赛被筛掉退役了的,好可惜。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/11/9
articleCard.readMore

建站技术 | 我的链接页面是如何实现的?

前言 近期读了一位大佬的文章:谈谈独立博客友链 | 瓦解的生活记事,颇有感悟。 随即大改友链页面,以实现收藏和推荐的作用,你也可以访问 失迹的博客 - 链接 默认样式 初次修改后 最终样式 主要的改动点在于: 两栏修改为更紧凑的三栏 区分板块 允许在链接右下角加入一个小图标(我称之为 “badge”)以增加一个链接 适配了暗黑模式和移动端 下面我会给出我的代码,并简单讲解作用。 由于我只针对我的博客,所以不保证能在其他地方使用,但思路必然可以参考。 实现代码 SCSS assets\scss\custom.scss: //链接页面 @media (min-width: 1024px) { .article-list--compact.links { display: grid; grid-template-columns: 1fr 1fr 1fr; background: none; box-shadow: none; article { background: var(--card-background); border: none; box-shadow: var(--shadow-l2); margin-bottom: 8px; border-radius: 10px; &:not(:nth-child(3n)) { margin-right: 8px; } } } } .article-list--compact.links { margin-top: 10px; margin-bottom: 40px; article { position: relative; .badge { position: absolute; width: 40px; height: 40px; line-height: 40px; border-radius: 100%; bottom: 10px; right: 10px; padding: 0 !important; display: block !important; text-align: center; font-size: 20px; background-color: var(--card-background); box-shadow: var(--shadow-l2); } } } .article-list--title { color: var(--accent-color); font-weight: 700; font-size: 1.6rem; margin-left: 10px; } 具体而言,针对电脑端我们开启了三栏显示,同时增加了我们新增元素的相关样式。 json 当然,我们需要补充更多的数据且格式有所变化。 data\links.json: [ { "title": "失迹的网站", "links": [ { "title": "失迹的博客", "website": "https://blog.reincarnatey.net/", "image": "korita.png", "description": "直至雨凝结之时:一个记录技术与思考的博客。" }, ... ] }, { "title": "博客聚合网站", "links": [ { "title": "开往 Travellings", "website": "https://www.travellings.cn/", "image": "travelling-light.png", "description": "「开往 Travellings」是一个友链接力项目,旨在通过网络跳转的方式将流量引入那些鲜为人知的独立站点,从而推动网络的开放性和多元性。", "badge": { "icon": "🚇", "link": "https://www.travellings.cn/go.html", "description": "跳转到随机博客" } },{ "title": "中文独立博客列表", "website": "https://github.com/timqian/chinese-independent-blogs", "image": "", "description": "" }, ... ] }, ... ] html layouts\page\links.html: {{ range $i, $category := $.Site.Data.links }} <span class="article-list--title">{{ $category.title }}</span> <div class="article-list--compact links"> {{ $siteResources := resources }} {{ range $i, $link := $category.links }} <article> <a href="{{ $link.website }}" target="_blank" rel="noopener"> <div class="article-details"> <h2 class="article-title"> {{- $link.title -}} </h2> <footer class="article-time"> {{ with $link.description }} {{ . }} {{ else }} {{ $link.website }} {{ end }} </footer> </div> {{ if $link.image }} {{ $image := $siteResources.Get (delimit (slice "link-img/" $link.image) "") | resources.Fingerprint "md5" }} {{ $imageResized := $image.Resize "120x120" }} <div class="article-image"> <img src="{{ $imageResized.RelPermalink }}" width="{{ $imageResized.Width }}" height="{{ $imageResized.Height }}" loading="lazy" data-key="links-{{ $link.website }}" data-hash="{{ $image.Data.Integrity }}"> </div> {{ end }} </a> {{ if $link.badge }} <a href="{{ $link.badge.link }}" target="_blank" title="{{ $link.badge.description}}" rel="noopener" class="badge">{{ $link.badge.icon }}</a> {{ end }} </article> {{ end }} </div> {{ end }} 由于这一部分改动略大,所以直接一整段放上来了。 大致上就是改成了对应的布局。 附录 参考文献 本人 css 不好,多次请教了 ChatGLM 一些有关 css 的教程或文档 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/8/3
articleCard.readMore

学习笔记 | Cloudflare 邮箱收到邮件时转发至多个邮箱

起因 尽管官方提供的功能已经很足,但总有人会产生新的需求。 像我,这里需要使 Cloudflare 邮箱收到邮件时同时转发到数个邮箱。 官方并不提供这个功能,因此我们需要使用电子邮箱 Worker 实现。 实现 在”电子邮件路由“页面创建一个电子邮件 Worker,写入以下代码: export default { async email(message, env, ctx) { await message.forward("email@example.com"); await message.forward("email2@example2.com"); } } 当然你还可以继续添加更多个,但理论上来说转发到的邮箱都需要在“目标地址”中经过验证。 然后在“路由规则”中添加到 Catch-all 或者某个自定义地址。 之后可以给自己发送一封邮件进行测试,可以看到邮件成功被转发到了多个邮箱。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/8/3
articleCard.readMore

建站技术 | 如何让 Bing 必应收录你的网站图标

问题描述 网页成功被 Bing 搜索引擎收录,但图标却没有被收录。在 Bing 中搜索自己的网页,在网站的左侧无法显示图标,而是显示一个灰色小地球,别人的网站左侧却可以正常显示图标。 由于我之前的截图找不到了,所以对比一下其他的网站 问题的原因在于,Bing 没有收录你的网站的图标。 准备 想要 Bing 收录你的图标,首先你的网站应该有一个图标,并具备特定程度的资质。 事先声明 本文章分享了我所使用并成功的方法,但我无法保证本方法是正确的收录图标的方法,也不保证本方法在未来依旧有效。 但目前除此方法之外,据说只能等待资质达到标准后 Bing 主动抓取收录图标。 关于资质 这一点具体标准并不明确,可以确定的是显然你的网站需要被 Bing 收录,但若你的网站被收录的时间较长,内容较多且质量较高,相对来说我认为收录的概率会大一些。 这里放上我申请时的资质: 收录时间超过一年 超过二十篇原创文章 Bing 近半年点击次数 1.6k 由于具体标准未知,所以大家都可以尝试,欢迎在评论区分享你的资质与结果,为他人提供参考! 正确的添加你的图标 正确的标准依旧未知,但显然你至少需要用某种方式添加你的图标: <link rel="shortcut icon" type="image/x-icon" href="/favicon.ico"> 推荐将图标文件放置在网站根目录。 提交 support request 在 Bing Webmaster Tools 中点击右上角的问号,点击"必应网站管理员支持"。 进入支持页面 进入支持页面后,点击页面下方蓝色按钮 “Raise support request”。 发起 support request 填写表格,网站选择你申请收录的网站,问题类型选择 “Crawling issue”,再选择 “Other”,并描述你的问题(可以按照我申请时写的 “Bing didn’t crawl my site’s icon”),最后填写验证码,点击蓝色按钮 “Get Help”。 填写信息 完成提交后,你会立即收到两封邮件,分别是你提交的信息以及官方收到你的提交。 后续 关注邮箱等待回复即可。 这里我在申请后的第三天便收到了回复: I would like to inform you that the favicon has been updated from our end and it will take 7-8 days to reflect on the Bing search results. 表示已经成功更新我的网站图标,大约在一周后便可以在搜索引擎中看到。 然而实际上当天就已经可以在搜索引擎中看到图标了,详见 随笔杂谈 | 博客二三事。 实际需要的时间也并不确定,例如:我申请的那一天是周五,收到的那天是周一,我等待三天可能与跨了一个周末有关。目前根据评论的反馈,最快的貌似一天就收录了。 你需要的时间可能更短也可能更长,欢迎尝试后在评论区分享你收到回复的用时以供参考。 写在最后 没想到困扰我这么久的方法这么轻松就解决了,以前怎么就没找到这么个入口呢,网上也搜索不到相关内容。 另外,如果你跟着我的文章进行了尝试,无论结果如何,欢迎在评论区分享你的相关情况(例如资质、用时、结果)以供更多人参考! 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/8/2
articleCard.readMore

踩坑记录 | Windows 删除文件/文件夹提示“找不到该项目”

问题描述 删除文件夹时,发现其中存在一个顽固文件,不管怎么删除都会提示”找不到该项目“,无法删除。 根据网络所见,貌似文件夹也存在这种情况。 问题猜想 我这里存在的问题推测是因为文件名末尾多带了一个 .,即类似于 example.file.txt.,正常操作是无法创建这样的文件的,貌似我这里是由于旧版本 adb 的 bug 所产生。 该文件并非无拓展名,具体肯定是属于异常情况,但这里不深究。 据说文件夹也存在这样的异常。 解决方法 方法一 这里提供我试验过后的成功方法。 创建一个 del.bat: del /f /a /q \\?\%1 rd /s /q \\?\%1 然后将要删除的东西拖到这个批处理文件上即可。 我这里直接拖入顽固文件不起作用,但是将顽固文件所在的文件夹直接拖入就删除成功了。 如果连正常文件都无法起作用,重新创建批处理文件并将编码修改为 ANSI。 方法二 右键,选择 Winrar 的添加到压缩文件,并勾选删除源文件(或者叫”压缩后删除原来的文件“)。 我这里直接对顽固文件进行操作未能成功,没有对所在文件夹尝试过,一些网友说有用,故也记录下来。 附录 参考文献 Windows桌面文件夹删除被提示“找不到该项目”——顽固目录、文件的删除_电脑桌面文件夹删除提示找不到该项目-CSDN博客 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/29
articleCard.readMore

随笔杂谈 | 博客二三事

前言 今天是我的博客建站 555 天。近来不止博客,我的网站均发生了许多的变化以及事件,以这篇文章作简述。 我的博客 这段时间里,我的博客是变化与事件最多的,其中最大的事件莫过于解决了此前博客在大陆访问一直不通畅的问题,我从 Netlify 迁移到了 Vercel。 迁移前 迁移后 令人感慨。 剩余的变化与事件,简而言之: 更好的接入了 Waline,同时记录了过程发表了一篇文章 加入了十年之约、开往、个站商店、博友圈、中文独立博客列表等项目 将上述项目加入了友链,都是非常好的项目,也算是助了一份力 除了原本就有的 Bing 与 Google 外,新增百度、360、Yandex 搜索引擎收录 更新了关于、友链、计划页面,开始记录博客大事记 侧栏加入了开往的接力链接,欢迎随意探索,发现更多的博客 更换了我的 Github 个人主页链接,因为改了名 新增了 Robots.txt 解决了 Bing 搜索页面不显示网站图标的问题 尽力优化了 SEO,真的 我的其他网站 除此之外,失迹云也历经数次更新,虽然看上去变化不大,但是修复了一些较为严重的 bug。 同时,尽管我的个人主页原本在大陆的访问较为通常,不过我依旧选择从 Netlify 迁移到了 Vercel。 迁移前 迁移后 同时也类似于我的博客,尽力优化了 SEO,更换了 Github 链接,新增了 Robots.txt,也顺带解决了网站图标的事。 未来数月内的计划 接下来一段时间内不会有新文章更新,也不会有较大的改动,大约半个月后会进行一次较集中的更新。 此后内容更新会变得较为缓慢,字数可能不长,内容可能简单,但质量不会下降(我认为的)。 遇到的一些有趣的事情 按时间顺序排序。 关于第三方 Cookie 弃用(aka. 3PCD) 谷歌,你先看看自家 Analytics。 谷歌只需要在前方 3PCD 就可以了,而使用 GA 的博主要考虑的就多了 你们要不吵一架?根据官方仓库下的 issue,目前该问题谷歌内部正在商讨解决方案。 在 Cloudflare 配置里看到一个有意思的选项。 笑死我了 我承认我的笑点很奇怪,但是本质上为关和 I’m under attack 真的很好笑。 等博客更新以后我要写在那篇文章最开头。 被 Waline 开发者巡回了 困扰了我许久的网站图标不收录问题,提了个工单四天就解决了。 个人主页 博客 考虑到之前在网络上搜索不到可以参考的解决方案,之后或许会单独发一篇博客来分享我的解决方法。 写完这篇文章之后已经到第二天了,所以本文的第一句话就存在问题。 附录 参考文献 随笔还有参考文献? 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/22
articleCard.readMore

建站技术 | 使博客更好地接入 Waline

前言 近期为我的网站进行了一次整体的升级,修复了个人主页和云盘的诸多 bug,新增了更多功能。其中包括优化 Waline 的使用,在这里记录下来以供他人参考。 这里我使用的是 Hugo 以及 stack 主题,使用其他方式的可以更多的参考 waline 的官方文档。 迁移到最新版本 Waline 客户端 由于 stack 模板中所接入的 waline 版本为 V2,或者像我一样从更远古的版本就开始魔改的话引入的是 V1,因此这里我们先来迁移到 V3 版本。 直接按照官方文档中的引入方式修改代码即可,这里贴上我将 stack 模板修改后的代码。 layouts\partials\comments\provider\waline.html: <link rel="stylesheet" href="https://unpkg.com/@waline/client@v3/dist/waline.css"/> <div id="waline" class="waline-container"></div> <style> .waline-container { background-color: var(--card-background); border-radius: var(--card-border-radius); box-shadow: var(--shadow-l1); padding: var(--card-padding); --waline-font-size: var(--article-font-size); } .waline-container .wl-count { color: var(--card-text-color-main); } </style> {{- $showReaction := (default true .Params.reaction) -}} {{- with .Site.Params.comments.waline -}} {{- $config := dict "el" "#waline" "dark" `html[data-scheme="dark"]` -}} {{- $replaceKeys := dict "serverurl" "serverURL" "requiredmeta" "requiredMeta" "wordlimit" "wordLimit" "pagesize" "pageSize" "imageuploader" "imageUploader" "texrenderer" "texRenderer" "commentsorting" "commentSorting" "recaptchav3key" "recaptchaV3Key" "turnstilekey" "turnstileKey" -}} {{- $replaceLocaleKeys := dict "reactiontitle" "reactionTitle" "gifsearchplaceholder" "gifSearchPlaceholder" "nickerror" "nickError" "mailerror" "mailError" "wordhint" "wordHint" "cancellike" "cancelLike" "cancelreply" "cancelReply" "uploadimage" "uploadImage" -}} {{- range $key, $val := . -}} {{- if ne $val nil -}} {{- $replaceKey := index $replaceKeys $key -}} {{- $k := default $key $replaceKey -}} {{- if eq $k "locale" -}} {{- $locale := dict -}} {{- range $lkey, $lval := $val -}} {{- if ne $lval nil -}} {{- $replaceLKey := index $replaceLocaleKeys $lkey -}} {{- $lk := default $lkey $replaceLKey -}} {{- $locale = merge $locale (dict $lk $lval) -}} {{- end -}} {{- end -}} {{- $config = merge $config (dict $k $locale) -}} {{- else if eq $k "reaction" -}} {{- $config = merge $config (dict $k (cond $showReaction $val false)) -}} {{- else -}} {{- $config = merge $config (dict $k $val) -}} {{- end -}} {{- end -}} {{- end -}} <script type="module"> import { init } from 'https://unpkg.com/@waline/client@v3/dist/waline.js'; init({{ $config | jsonify | safeJS }}); </script> {{- end -}} 这里我使用的 VSCode 虽然安装了 Hugo Language and Syntax Support 插件,但依旧无法正确判断 Hugo 语言语法,倒数第三行会报错,只要能正常运行即可。 另外,除了引入 V3 以外,这里也包括了 locale 大小写的修复,后文会提到,以及在 FrontMatter 中新增了一个自定义属性 reaction,用以开关文章反应,默认为 true。 刷新网页,如果发现评论系统右下角变成了 Powered by Waline v3.3.0 就说明升级成功了。 如果你曾经已经接入 Waline 的更多功能例如统计,那么你可能需要注意按照官方的迁移指南修改你的代码。 服务端 由于数据库与服务端分开部署,所以升级服务端理所当然无需担心会丢失数据。 在管理端 UI 界面最上方可以看到当前最新版本为 @waline/vercel@1.32.0。 官方给的方式是将自己的 waline github 库中的 package.json 中的 @waline/vercel 版本修改为最新的即可。当然理论上来说,我们这里用的是 latest,直接在 Vercel 控制台中重部署即可。 我这里版本差的有点多,所以选择将自己的库 pull 下来,然后替换成官方的 waline/example at main · walinejs/waline · GitHub 再 push 回去。 接着回到 Vercel 控制台,看到已经重新部署了,打开管理端也可以看到上方不再显示版本更新提示。 关于 Vercel 墙内 Dns 污染 这里顺带一提,污染会导致墙内无法访问,进而墙内无法加载 Waline。 解决方法:自定义域名,并且 CNAME 换成 cname-china.vercel-dns.com。 本身访问速度已经较快了,建议关掉自定义域名 Cloudflare 代理,不然会出现 Cloudflare 的代理 ip 地址。 配置邮件提醒 本步骤需要修改较多的环境变量,你可以在这里查看相关的服务端环境变量配置的详细信息:服务端环境变量 | Waline。 配置 Gmail 邮箱 首先需要对谷歌邮箱账号进行一些设置。 在邮箱设置的转发和 POP/IMAP 中启用 IMAP。 启用 IMAP 然后在这里生成谷歌账号的应用专用密码,复制下来一会用。 随后按照官方指南 评论通知 | Waline 配置 Vercel 上的环境变量即可: SMTP_SERVICE:填 Gmail SMTP_USER:填谷歌邮箱 SMTP_PASS:填刚刚获取的应用专用密码,有 16 位,记得删除空格 SITE_NAME:填博客的名字 SITE_URL:要带 https://,不带最后一个斜杠,例如 https://blog.reincarnatey.net AUTHOR_EMAIL:填自己常用的邮箱,用来接受评论通知 套上自己的 Cloudflare 邮箱 当然直接用谷歌邮箱发还是过于粗糙,这里希望再套上自己的 Cloudflare 邮箱实现更好的效果。这里的操作就类似于创建 Cloudflare 邮箱然后绑定到谷歌邮箱的别名上,详细可以在网络上搜索相关博客,这里只简单记录一下步骤。 在 Cloudflare 电子邮件路由设置中创建邮箱地址,需要这一步是因为待会要收一个验证邮件(当然可以像我一样设置 Catch-All 就不用这一步) 谷歌账号需要开启一些设置(例如开启双重验证) 像刚刚一样再创建一个应用专用密码 在谷歌邮箱账号设置中在账号与导入页面选择添加其他电子邮件地址 第一步名称和电子邮件地址填写你要导入的名称和电子邮件地址,例如 失迹的博客 和 noreply@reincarnatey.net,勾选作为别名 SMTP服务器:填 smtp.gmail.com 用户名:填你的谷歌邮箱 密码:填刚刚获取的应用专用密码,记得删空格 别名电子邮件会收到一封邮件,完成验证 添加 Vercel 的环境变量为新的邮箱信息,与在谷歌那边填的一致即可 SENDER_NAME:自定义发件人名字,如 失迹的博客 SENDER_EMAIL:自定义发件地址,如 noreply@reincarnatey.net 这样就完成别名的设置了,如果要精致一点还可以为这个邮箱设置一个 gravatar。 我们发送一个回复,可以看到收到的邮件的发件人信息已经设置好了。 评论提醒邮件 使用模板样式 这里我使用的模板修改自 SaraKale’s blog,非常感谢。 效果展示: 访客评论被回复收到的 站长收到的 MAIL_SUBJECT:{{parent.nick | safe}},您在 『{{site.name | safe}}』 上的评论收到了回复 MAIL_TEMPLATE: <div style="background: url(https://tva3.sinaimg.cn/large/c56b8822ly1h62npb7s1ej201y01y0lh.jpg);padding:40px 0px 20px;margin:0px;background-color:#FFCDCE;width:100%;"> <style type="text/css"> @media screen and (max-width:600px) { .afterimg, .beforeimg { display: none !important } } </style> <div style="border-radius: 10px 10px 10px 10px;font-size:14px;color: #555555;width: 530px;font-family:'Century Gothic','Trebuchet MS','Hiragino Sans GB',微软雅黑,'Microsoft Yahei',Tahoma,Helvetica,Arial,'SimSun',sans-serif;margin:50px auto;max-width:100%;background: ##ffffff;"> <img class="beforeimg" style="width:530px;height:317px;z-index:-100;pointer-events:none" src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/before.png"> <img src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/violet.jpg" style="width:100%;overflow:hidden;pointer-events:none;margin-top: -120px;"> <div style="width:100%;background:#f8d1ce;color:#9d2850;background-image: -moz-linear-gradient(0deg, rgb(67, 198, 184), rgb(255, 209, 244));height: 66px;background: url(https://tva2.sinaimg.cn/large/c56b8822ly1h61tb7tagcj20ii01u3yc.jpg) left top no-repeat;display: flex;justify-content: center;flex-direction: column;"> <p style="font-size:16px;font-weight: bold;text-align:center;word-break:break-all;margin:0;"> 您在<a style="text-decoration:none;color: #9d2850;" href="{{site.url}}">『{{site.name | safe}}』</a>上的留言收到一条回复</p> </div> <div class="formmain" style="background:#fff;width:100%;max-width:800px;margin:auto auto;overflow:hidden;margin-bottom: -155px;"> <div style="margin:40px auto;width:90%;"> <p>😊Hi,{{parent.nick}},您曾发表的评论:</p> <div style="background: #fafafa repeating-linear-gradient(-45deg,#fff,#fff 1.125rem,transparent 1.125rem,transparent 2.25rem);box-shadow: 0 2px 5px rgba(0, 0, 0, 0.15);margin:20px 0px;padding:15px;border-radius:5px;font-size:15px;color:#555555;"> {{parent.comment | safe}}</div> <p><strong>{{self.nick}}</strong> 回复您:</p> <div style="background: #fafafa repeating-linear-gradient(-45deg,#fff,#fff 1.125rem,transparent 1.125rem,transparent 2.25rem);box-shadow: 0 2px 5px rgba(0, 0, 0, 0.15);margin:20px 0px;padding:15px;border-radius:5px;font-size:15px;color:#555555;"> {{self.comment | safe}}</div> <p><a style="text-decoration:none; color:#cf5c83" href="{{site.postUrl}}" target="_blank"> 查看回复的完整內容 </a>,欢迎再次访问<a style="text-decoration:none; color:#cf5c83" href="{{site.url}}" target="_blank"> {{site.name}} </a>。 <hr /> <p style="font-size:14px;color:#b7adad;text-align:center;position: relative;z-index: 99;"> 本邮件为系统自动发送,请勿直接回复邮件哦,可到博文内容回复。<br />{{site.url}}</p> </p> <img src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/line.png" style="width:100%;margin:25px auto 5px auto;display:block;pointer-events:none"> <p class="bottomhr" style="font-size:12px;text-align:center;color:#999">© 失迹的博客</p> </div> </div> <img class="afterimg" style="width:535px;height:317px;z-index:100;margin-left: -3px;" src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/after.png"> </div> </div> MAIL_SUBJECT_ADMIN:您有一条新的评论 MAIL_TEMPLATE_ADMIN: <div style="background: url(https://tva3.sinaimg.cn/large/c56b8822ly1h62npb7s1ej201y01y0lh.jpg);padding:40px 0px 20px;margin:0px;background-color:#FFCDCE; width:100%;"> <style type="text/css"> @media screen and (max-width:600px) { .afterimg, .beforeimg { display: none !important } } </style> <div style="border-radius: 10px 10px 10px 10px;font-size:14px;color:#555555;width: 530px;font-family:'Century Gothic','Trebuchet MS','Hiragino Sans GB',微软雅黑,'Microsoft Yahei',Tahoma,Helvetica,Arial,'SimSun',sans-serif;margin:50px auto;max-width:100%;background: ##ffffff;"> <img class="beforeimg" style="width:530px;height:317px;pointer-events:none" src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/before.png"> <img src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/violet.jpg" style="width:100%;overflow:hidden;pointer-events:none;margin-top: -120px;"> <div style="width:100%;background:#f8d1ce;color:#9d2850;background-image: -moz-linear-gradient(0deg, rgb(67, 198, 184), rgb(255, 209, 244));height: 66px;background: url(https://tva2.sinaimg.cn/large/c56b8822ly1h61tb7tagcj20ii01u3yc.jpg) left top no-repeat;display: flex;justify-content: center;flex-direction: column;"> <p style="font-size:16px;font-weight: bold;text-align:center;word-break:break-all;margin:0;"> 您在 <a style="text-decoration:none;color:#9d2850;" href="{{site.url}}" target="_blank">{{site.name}}</a> 上的文章有了新的评论</p> </div> <div class="formmain" style="background:#fff;width:100%;max-width:800px;margin:auto auto;overflow:hidden;margin-bottom:-155px;z-index:100;"> <div style="margin:40px auto;width:90%;"> <p><strong>{{self.nick}}</strong> 发表了评论:</p> <div style="background: #fafafa repeating-linear-gradient(-45deg,#fff,#fff 1.125rem,transparent 1.125rem,transparent 2.25rem);box-shadow: 0 2px 5px rgba(0, 0, 0, 0.15);margin:20px 0px;padding:15px;border-radius:5px;font-size:15px;color:#555555;"> {{self.comment | safe}}</div> <p style="text-align:center;position: relative;z-index: 99;"><a style="text-decoration:none;color:#cf5c83" href="{{site.postUrl}}" target="_blank">查看回复的完整內容</a> </p> <img src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/line.png" style="width:100%;margin:25px auto 5px auto;display:block;pointer-events:none"> <p class="bottomhr" style="font-size:12px;text-align:center;color:#999">© 失迹的博客</p> </div> </div> <img class="afterimg" style="width:535px;height:317px;margin-left:-3px;" src="https://npm.elemecdn.com/hexo-butterfly-envelope/lib/after.png"> </div> </div> 直接在 Vercel 设置环境变量即可,大胆复制进去。注意如果选择压行的话会丢掉一部分空格,需要手动补上否则不美观。 更多的 Waline 客户端配置 可以在官方文档 组件属性 | Waline 中查看。 放上这里我的配置: waline: serverURL: https://waline.reincarnatey.net/ lang: zh-CN search: false pageview: true comment: true emoji: - https://npm.elemecdn.com/@waline/emojis@1.1.0/bmoji - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili - https://npm.elemecdn.com/@waline/emojis@1.1.0/weibo reaction: - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili/bb_heart_eyes.png - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili/bb_thumbsup.png - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili/bb_zhoumei.png - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili/bb_grievance.png - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili/bb_dizzy_face.png - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili/bb_slap.png requiredMeta: - nick locale: admin: 站长 sofa: 还没有人评论哦!快来抢沙发吧~ placeholder: 欢迎留下宝贵的评论!请留下正确的邮箱以便有回复时进行邮箱提醒,请勿发布任何与本文章无关的内容。 reactionTitle: 这篇文章对你有帮助吗? reaction0: 非常有用 reaction1: 有帮助 reaction2: 一般 reaction3: 无帮助 reaction4: 看不懂 reaction5: 有错误 需要注意的是由于 yaml 读取为统一小写,而 Waline 的配置又大小写敏感,所以需要专门的步骤将大小写转换。 我在前文提供的 layouts\partials\comments\provider\waline.html 已包含了基本配置以及 locale 配置的大小写修复,其他配置同理。同时也包含新增了一个文章 FrontMatter 属性用以控制是否显示文章反应:reaction: true。 接入浏览量与评论数统计 文章页 在 config.yaml 的 waline 配置中将 pageview 与 comment 都设置为 true。 并且修改 layouts\partials\article\components\details.html,将末尾的 <footer> 中的内容修改为: <footer class="article-time"> {{ if not .Date.IsZero }} <div> {{ partial "helper/icon" "date" }} <time class="article-time--published"> {{- .Date.Format (or .Site.Params.dateFormat.published "Jan 02, 2006") -}} </time> </div> {{ end }} {{ if .Site.Params.article.readingTime }} <div> {{ partial "helper/icon" "clock" }} <time class="article-words"> {{ $fixedWordCount := add .WordCount 224}} 约{{ div $fixedWordCount 225 }}分钟,共{{ .WordCount }}字 </time> </div> {{ end }} <div> {{ partial "helper/icon" "view" }} <time class="article-pageview"> <span class="waline-pageview-count" data-path="{{.RelPermalink}}">0</span> 次浏览 </time> </div> <div> {{ partial "helper/icon" "comment" }} <time class="article-comment"> <span class="waline-comment-count" data-path="{{.RelPermalink}}">0</span> 条评论 </time> </div> </footer> 另外虽然有提供 .ReadingTime 变量以供估算阅读时长,但官方设置的速度或许是基于英文的,个人感觉误差较大。所以这里根据网络上的数据,成人每分钟阅读约 200-250 字进行向上取整估算。 至于图标,按照这里所使用的方式下载对应的图标 view.svg 和 comment.svg,并修改成适配暗黑模式后,扔到 \assets\icons 下即可。 由于文章页会直接引入完整的 Waline,所以不需要额外的引入了。可以看到在文章页面已经可以正常显示修改的内容了。 显示浏览量与评论数 页脚 这里希望在所有页面的页脚展示主页的浏览量。 由于除了文章以外的页面不会引入浏览量统计,所以这里需要额外引入,在 layouts\partials\footer\footer.html 的 <footer> 内添加: <script type="module"> import { pageviewCount } from 'https://unpkg.com/@waline/client@v3/dist/pageview.js'; pageviewCount({ serverURL: "{{.Site.Params.comments.waline.serverURL}}" }); </script> 然后在页脚相关代码中,你需要展示的地方添加上显示的 span 即可,例如在某个地方添加: 共 <span class="waline-pageview-count" data-path="/">0</span> 次浏览 可以看到各页面都可以成功显示首页浏览量,同时这里悄悄解决了首页中浏览量显示的问题。 首页浏览量 这里也可以测试,首页浏览量:0 首页浏览量:<span class="waline-pageview-count" data-path="/">0</span> 首页 由于首页不开启评论功能,而页脚中只引入了浏览量统计,所以首页文章列表中显示不了评论数。我们希望能够显示出文章列表中各文章的评论数,需要额外的方式进行引入。 在 layouts\index.html 中 </section> 后面加上: <script type="module"> import { commentCount } from 'https://unpkg.com/@waline/client@v3/dist/comment.js'; commentCount({ serverURL: "{{.Site.Params.comments.waline.serverURL}}" }); </script> 可以看到主页文章列表中,浏览量与回复数统计数据已经正常填充。 接入最新评论挂件 组件效果(自定义样式) 可以在 客户端 API | Waline 页面查看这个挂件的 API。 这里我们以侧边栏组件的形式引入这个挂件,并设计成只有首页会出现。 在 assets\scss\partials\widgets.scss 的最下方添加: /* Recent Comment widget */ .widget.recentComment { .recent-comment-list { border-radius: var(--card-border-radius); box-shadow: var(--shadow-l1); background-color: var(--card-background); } .recent-comment-list-item { display: flex; padding: 0 1.25em; align-items: center; &:not(:last-of-type) { border-bottom: 1.4px solid var(--card-separator-color); } } .recent-comment-card { flex: 1; width: 0; padding: 20px 10px !important; p { font-size: .8em; margin-top: 3px; margin-bottom: 0px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; color: var(--card-text-color-secondary); } } .recent-comment-avatar { width: 3rem; border-radius: 50%; } .recent-comment-nick { font-size: .875em; color: var(--card-text-color-main); } .recent-comment-time { font-size: .75em; color: var(--card-text-color-tertiary); } .vemoji, .wl-emoji { height: 1.25em; margin: -0.2em .1em; } } 创建文件 layouts\partials\widget\recent-comment.html: {{ if .IsHome }} <section class="widget recentComment" hidden> <div class="widget-icon"> {{ partial "helper/icon" "comment" }} </div> <h2 class="widget-title section-title">最新评论</h2> <div class="recent-comment-list"></div> </section> <script type="module"> import { RecentComments } from 'https://unpkg.com/@waline/client@v3/dist/waline.js'; RecentComments({ serverURL: "{{.Site.Params.comments.waline.serverURL}}", count: 5, }).then((comments) => { if (comments.comments.data.length) { document.getElementsByClassName('recentComment')[0].removeAttribute("hidden"); document.getElementsByClassName('recent-comment-list')[0].innerHTML = comments.comments.data.map((comment) => `<a class="recent-comment-list-item" href="${comment.url}" target="_blank"> <img class="recent-comment-avatar" src="${comment.avatar}"/> <div class="recent-comment-card"> <div> <span class="recent-comment-nick">${comment.nick}</span> <span class="recent-comment-time">${new Date(comment.time).toISOString().split('T')[0]}</span> </div> <p>${comment.comment.replaceAll("<br>", " ").replaceAll("\n", " ").replace(/<\/?a.*?>/g, "")}</p> </div> </a>` ).join(''); } }); </script> {{ end }} 当然我们还需要在 config.yaml 中开启这个组件: params: widgets: enabled: - recent-comment # 加上这一条 最外面那一个 if 就是用来限制只在首页出现的。这里设定成获取到数据后再显示,以及将评论内容中的链接全部去掉了,设定成展示最近的 5 条评论。 由于官方样式令人堪忧,所以这里是获取数据后自定义。关于 comment 的属性,可以在 API | Waline 中查看关于 BaseWalineResponseComment 的接口,这里简单介绍几个比较有用的: 属性 类型 含义 说明 avatar string 头像链接 comment string 评论内容 渲染成 html 后的 like number 评论喜欢数 link string 用户地址 用户自己填写的,不建议用,避免出现乱七八糟的 backlink 和影响 SEO nick string 用户昵称 用户自己填写的 orig string 原始评论 time number 评论时间 时间戳,单位毫秒 url string 原文章地址 当然,这里没有考虑到国际化的需求以及更多的可扩展性,欢迎有想法的朋友实现更高拓展性和更美观的侧边栏组件。 后续计划再更新一次,添加显示评论所在的原文章,会在之后关于侧边栏的文章中再次提及。 除此之外 Waline 还有一个 UserList 挂件,我这里并不需要,要接入的话也是类似的。 结语 至此,我们已经捣鼓完了 Waline 大部分功能。你可以在官方手册中了解到更多的配置项以及我未提及的插件系统。 很遗憾此前我曾对模板进行过一次较大的改动,实现了例如独立出来的 Solutions 题解区以及 Algorithms 算法标签等,由于过于久远已经不记得调整内容无法写成文章,同时也因为放弃了拓展性的修改,无法合适地将我修改过后的主题开源。 当然,对博客主题的改造还远不止于此。后续的改造我也会保持记录,并发布文章,欢迎在评论区插眼或是订阅 RSS 以便关注后续更新。 有不足之处请见谅,有错误欢迎指正,也欢迎大家交换友链或是在评论区玩耍~ 附录 参考文献 Walin 官网 使用Waline给Hexo静态博客添加评论系统 | Quantum Bit 评论插件 Waline 之邮件通知配置 | Oragekk’s Blog Gmail发送邮件的配置方法_gmail smtp-CSDN博客 waline邮件通知模板样式一览 | SaraKale’s blog 函数 | Hugo官方文档 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/19
articleCard.readMore

学习笔记 | OneIndex 伪静态配置以及迁移到 PHP8.2

伪静态配置 不知道你是否存在该问题:开启伪静态后,访问文件时若目录或文件名带有空格则无法浏览和下载。 如果我们手动加上 /?/ 则可以正确访问,那么显然这是伪静态的锅。然而以我的正则水平完全看不出我的伪静态为何有问题,遂求助网络,发现只有这篇博文中提供的伪静态文件可以解决以上问题: AddDefaultCharset utf-8 Options +FollowSymLinks IndexIgnore */* RewriteEngine on # if a directory or a file exists, use it directly RewriteCond %{REQUEST_FILENAME} !-f RewriteCond %{REQUEST_FILENAME} !-d # otherwise forward it to index.php RewriteRule . index.php 很神奇,以我的正则水平实在看不懂为何这是有效的,但正所谓能跑就行。 迁移到 PHP8.2 最近我的盘又出问题了,发现输出了一堆错误,所以这里总结了一下自从升级到 PHP8.2 后遇到的错误时能参考的博客: 踩坑记录 | OneIndex 首页出现 HTTP ERROR 500 (转载)PHP 8.0 部分错误的解决办法 - 无主界 (Deprecated: Required parameter $xxx follows optional parameter $yyy ) PHP8中使用$_POST或者$_GET出现Undefined array key “xx“ 总而言之,不需要改动过多代码,只需要在报错的地方删除或者加上一些 isset 即可。但尤其需要注意部分 if 中 isset 的逻辑,避免产生漏洞。 附录 参考文献 搭建oneindex作为外链/图床 - moluuser’s Blog 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/18
articleCard.readMore

踩坑记录 | Windows 下 PHP 开启 curl 扩展

前言 相信各位一定搜到过许多方法,什么移动到 Win32 、修改 apache 加载、手动安装 curl 什么的。 由于我并非专业后端开发,只是想跑一跑代码,因此并没有安装 apache,这时候如何为 PHP 开启 curl 拓展呢。 开启方法 首先修改 php.ini,将 extension=curl 前面的分号去掉。 但是在 cmd 中输入 php -i 还是看不到开启了 curl 扩展,运行代码还会有以下报错信息: Warning: PHP Startup: Unable to load dynamic library 'curl' (tried: C:\php\ext\curl (找不到指定的模块。), C:\php\ext\php_curl.dll (找不到指定的模块。)) in Unknown on line 0 注意到报错信息里的拓展位置不对,我这里 PHP 根目录明明是在 D:\PHP8.2,因而找到解决办法。 修改 php.ini 的这一条配置: extension_dir = "D:\PHP8.2\ext" 再次查看 phpinfo,可以看到成功开启 curl 拓展,正常运行。 附录 参考文献 网上搜了一堆,甚至问过 Copilot,没一个有用的。 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/18
articleCard.readMore

学习笔记 | QQNT 聊天记录导出

更新 已将研究成果整合至文档: QQDecrypt:QQ 聊天数据库解密。 前言 不管是出于希望将聊天记录保存到本地,还是需要导出和将聊天记录用于语言模型训练等用途,都需要导出 QQ 聊天记录这一步。 近期发现 QQNT 也存在导出聊天记录的方法,遂进行尝试和记录,感谢各位前辈的探索和作出的贡献。 环境及准备工作 我的环境是 Windows10,QQNT 9.9.11-24568,Python 3.10.6。 需要任意一款反编译软件,以及用于查看数据库的 DB Browser for SQLite(安装包来自网络,可从参考文献第一条中获取链接)。 操作的所有文件建议在操作前进行备份。 逆向获取数据库密码 由于本人不会逆向,是按照参考文献第一条中的记录进行操作,故在此不重复记录。 需要注意的是附加到的进程是从上往下的第一个,与参考博客中的描述并不一致,也许不同设备会有不同。 如果进程选择正确,在附加后需要一定时间加载,且过程中有弹出窗口,并在 F9 继续运行之前无法使用 QQ,并在登录后触发断点。如果进程选择错误则以上会有不同,关掉 QQ 重新附加即可。 从数据库中导出数据 依旧按照参考文献第一条中的记录进行操作,不重复记录。 其中 nt_msg.header.txt 使用文本编辑器打开后可以在文件开头看到 SQLite header 3 以及 QQ_NT DB、HMAC_SHA1 等字样。亲测使用笔记本打开会显示乱码,而使用 VS Code 打开没有问题。 在 DB Browser for SQLite 中,打开菜单->工具->设置加密,不输入密码直接点击 OK 以解除数据库加密。 以私聊数据表 c2c_msg_table 为例,这里介绍一部分关键数据。 列名 类型 含义 说明 40050 int 发送时间 单位为秒的时间戳 40058 int 发送时间 单位为秒的时间戳,精确到天 40030 int 私聊对象 对方 QQ 号(无论是对方还是自己发送的消息) 40033 int 消息发送者 发送者的 QQ 号 40093 str 消息发送者 QQ 昵称或是备注名 40800 protobuf 消息内容 格式较为复杂,可使用 CyberChef 工具进行解析 关于 40800 的解析,由于较复杂所以不展开。 至此,已可以使用 sql 命令或编写程序导出数据。 附录 参考文献 GitHub - mobyw/GroupChatAnnualReport: 使用 QQNT Windows 聊天记录制作群聊年度报告! GitHub - Mythologyli/qq-nt-db: QQ NT Windows 数据库解密+图片/文件清理 GitHub - QQBackup/qq-win-db-key: 全平台 QQ 聊天数据库解密 解析 NTQQ 数据库 | 冷月的博客 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/7
articleCard.readMore

拾光纪行 | CEIC2024 游记

初赛 题目太面向对象了,打了一坨。 复赛 复赛题面就是一坨,题也是一坨,打了一坨。 你说得对,A 题是小学奥数题,但是我已经是高中生了,所以卡了半个多小时才想出来样例二。没想出来的时候写了个 dfs 验证一下,想出来之后感觉不会写索性就先交 dfs 后面再回来推逻辑了。但是没给数据范围,什么情况。 开 B,一眼 next_permutation 暴力,仔细算了算复杂度发现暴力变成正解了,随便写写就过掉了。 开 C,读题目挺简单的,一看样例发现看不懂,这和题目说的是一道题吗?遂不可以总司令跳过。 开 D,前面一堆废话,其实最后一句话就把题目说完了。刚准备开敲快速幂,一琢磨数据规模发现原来是高精度,遂转 py 打表。写了个 py 计算答案,再写了个 cpp 用快读的方式读入答案并把长度与前一百位打成表,于是本题 $O(1)$。然而悲催的是由于开题比较晚所以打表比较晚,数据范围是三万但是我 py 跑到一万九就差不多速度很慢了,然后我抽了一不小心关掉了输出没保存,于是重跑打表,只来得及跑到三千,遂只打了三千的表,亏爆。 赛后发现 B 题原来给的那个数字就是 $n$,服了,没说清楚,而且经同学暴力检验发现只要 $n\gt3$ 那么答案就只有 $4$ 和 $9$,$n\le3$ 就输出 $0$,不可以总司令在本题输给了输出样例。 唯一值得夸奖的比赛体验就是居家比赛,自家电脑上的 IDE 随便用。 结果:66 分(换算成 400 分制就是 264 分),喜提一等。 决赛 前言 8 月 2 号的时候想看证书发现晋级了,啊?思考了一会要不要去,毕竟又远又耽误课还不报销。 没想到就我这么烂的成绩也能进决赛,感慨。虽然深圳到乌镇有点远,但毕竟是全国的,难得有机会,所以还是报名了,也顺便去杭州和乌镇玩一玩。 关于旅游的部分,在这篇文章可以看到一些记录以及照片,本文章只记录有关比赛的内容。 游记 day -2 中午从宝安机场起飞,下午到萧山机场。然后坐了两个小时车到乌镇,等到了民宿之后已经是晚上了,所以在附近吃了个饭,然后在附近逛了逛,今天就结束了。 其实原定是一早起飞,然后下午到了还能去签到的,但临时有事所以改到中午了。 day -1 签到。 但由于原定昨天签到改成了今天,忘了签到地点有不同,因此一开始走错了地,看到那边大场馆已经在比赛了,还以为我们这项也是像 NOI 一样一大群人在大场馆比。 签到时使用的是二维码,然后发了一个包,包里有手册、比赛衣服以及参赛证之类的,除了手册以外都有纪念意义,还是很不错的。听负责签到的人说参赛时要穿着衣服然后带参赛证。 顺带一提,由于来这边比赛的人特别特别多,而且由于大部分赛项都是面向小学生初中生的因此还会拖家带口,所以基本到哪都能看见来参加比赛的人,甚至周边的出租车司机和店家之类的全都知道比赛的事情,很神奇。 day 0 当天下午稍晚一点到了考场,距离检录结束还有十多分钟。 跟随现场志愿者前往考场,中途差点进到初中组考场,后续爬了四楼到了高中组考场。进考场发现居然只有二十个人左右,并且我是最后一个进考场的,大概是跑远路参加决赛的不多。 赛前 随便找了一个座位坐下,发现其他选手已经开始敲键盘。显示屏极其古老,是很旧的方形的那种。印象中系统貌似装的是 win10,键盘还行。 需要打开官网登录账号,遂登录,输了很多遍密码都登录失败,诧异,遂打开记事本输入好后粘贴入框,发现这个机子的键盘绑键是错位的,我按 W 会打出来 Q,按 T 会打出来 R,但数字键盘是正常的,因此输入手机号时没有发现,但密码输入框是隐藏的,因此输入密码里的字母时就错了。 诧异,遂换机位,发现这一台的键盘莫得 F12,诧异,遂换机位,成功登录。 由于不是 NOI 系列比赛,不知道文件夹规则,因此没办法提前建好文件。编译器貌似只有比较经典的 dev cpp 和 vim 那些的,因此启动 dev cpp,还是英文版的,调了一些设置,然后敲了文件缺省,等开赛。 赛时 提前五分钟告知了比赛密码,在官网进入比赛并下载试卷。读题。 A 题读完之后第一个联想到的是 K Smallest Sums,虽然实际上关联不大,但也因此马上想到了一个重要性质,给出的最小的数字一定是由最小的两个加来的,因此这里可以作为一个枚举点,确定好前两个数之后就可以根据给出的第二小的数直接确定第三个数,然后一直递推下去,有多种可能只需要搜索一下。 此时认为会做 A,于是接着读 B,读完就会了。建一个表达式树然后 dp 一下即可,极其简单。成功切题,美滋滋。遂读 C,发现见过原题,但那道原题我不会做,悲。 此时比赛正式开始,两个小时敲两个会的题和两道暴力应该问题不大吧? 问题大得很。开始写之后才发现 A 的搜索难写的离谱。想不出来如何实现,然后浪费了一个小时。没错,比赛一半的时间完完全全浪费掉了。但是 A 题还是要写的,后面又花了半小时写了降一档的 next_permutation 暴力。 还剩半个小时,本来以为 B 已经会了还是能打的出来的,但很快发现了新问题:嘶,表达式树怎么建来着?哈哈,完蛋。想了整整二十分钟,只记得和笛卡尔树差不多,要用到栈什么的,半天写不出来。 还剩不到十分钟的时候放弃,输出样例了。然后 C 写了个 map 暴力,结果样例过不去,试着调暴力也没调出来,不理解,最后改成了 set 但是没测样例。D 输出样例。 最后一分钟极速删无关文件打包改文件名然后回官网提交,巨惊险,出现提交成功的页面的时候发现时钟已经 40 分,刷新一下就显示比赛结束了。 赛后 出考场之后,天很快就黑了。 比赛当晚的天空 进来时来不及拍照留念,出来的时候拍了许多。然后有一位老师一直在关心“孩子都出来没有”“找到家长了没有”哈哈哈。 对自己的表现不太满意,完全没有任何发挥,甚至都不确定代码有没有提交上去。 虽然是不太重要的比赛,但毕竟全国级的赛事只打出了这样的水平,还是很丢人的。 后日谈 当天晚上在酒店通过在官网上抓包,从一些请求的返回数据里面看到了自己提交的文件信息,这才确认代码成功卡着时间提交上去了,惊险,看来至少没有爆零。 以及查到了 C 的原题是可见的点,原来正解是欧拉筛求欧拉函数,幽默数论,不会。 毕竟第二天就是闭幕式和颁奖,所以理所当然第二天就出分了,具体分数很难看就不说了,总之打了一个 3=,突然有种打铜不如打铁的意味了,但丢人归丢人,但比起比赛其实这次体验更多的是出远门玩了好久,比赛带给我的遗憾情绪连那天晚上都没撑到就无影无踪了。 不过,接下来的时间会越来越快,越来越不够用,自然是不允许我再这样玩了,这篇游记也是我半周里每天的空隙敲出来的。不过我并没有感到遗憾,毕竟我一直对高考之后的事情有所期待。 我相信时机到了的时候,自然而然就该来到该有的状态,就像这篇文章就该于此停笔。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/7/7
articleCard.readMore

拾光纪行 | YIIC2024 游记

前言 实际上已经半 AFO 三个月了,再来参加这场比赛也没什么好说的,尽力而为吧,也无需过于在意。 这段时间洛谷上做题寥寥无几,只有偶尔会出一些题,比方说给学校高一集训出题和组题,或是给区里的竞赛(貌似是给初中生考的?)供题。 更多的还是回到 whk 上了。 初赛 初赛还是比较简单的,就不细讲了,补全代码都很好懂,不过阅读题有一篇只看出来树上背包,不知道合并的什么玩意儿,另外单选貌似也有选错的。 最后出分虽然是机房第一,但还是比我预想的八九十分要低一点点。另外这次高一学弟也晋级了三个,也有几个差了一点点,相比往年我们学校的情况感觉好多,但相比这届竞赛班人数又感觉有点少。 喜提一个市1=,虽然感觉作用不大。 复赛 总的来说,抛开题目不看,这次复赛是所有比赛中最舒服的一次。不管是比赛场地、设备、比赛系统、IDE、赛时安排之类的,都是体验最好的一次。 day -3 整理了一下官网给出的考试范围和对应的模版题。居然有网络流?而且复杂树和复杂图论里面也不知道有什么…数据结构也没说清楚。 这一天光整理题单去了,好像没做题。 day -2 敲了一些图论板子,只剩下欧拉回路和联通性相关的那些没敲了。 day -1 花了好多时间把三种联通分量和对应缩点、割点割边都重新复习和理解了一遍。也许我第一次学的时候没有理解清楚,一直有点模糊,这次终于区分清楚了。然后欧拉回路也重新理解了一遍。 发现这次重写模板题,又多了好几个优化代码减少码长的地方,学 OI 两年了技术也就这样,倒是压行和偷懒越来越会了。 网络流应该不考?不看了。数据结构还挺熟练的,不看了。字符串大纲说只考基础,不看了。 day 0 赛前 因为这天本来学校需要补高考假,所以是从学校出发。考点在广州的广技师,从深圳过去需要大概两个小时。 六点起床,大概 6:20 左右一边吃着面包一边赶到了门口。高一有两个学弟没在线请假,校门口刷脸没有开门权限,走人工通道要打电话跟教练确认,搞了有一会。高二另一位同学(之前一起打 THUPC 和省选的)不知道是起晚了还是怎么,来的有点晚,打电话催了一会。六点半多一点出发了。 洛谷运势大凶。在车上看一些骗分和防挂分的博客。 到了考场之后以为不用着急进去,上个厕所回来发现人都进去了,有人提醒我们这场比赛需要签到和登录?于是赶紧进考场了。 进来发现比赛场地环境好舒服,签完到之后找到自己座位坐下,发现比赛设备居然是笔记本电脑,而且提供的 IDE 除了常见的 devC 和 VSCode 以外居然还有小熊猫!震惊。听说要登录,不知所措,遂举手询问工作人员,原来要打开桌面某个程序登录。发现居然是 java 写的,会定时发卷和自动同步文件,我去这也太牛了。 发现周围的同学可以动电脑,遂启动小熊猫开始配置,调了一下编译选项和缺省模版之类的。在桌面上发现了样例文件夹,打开来看了一眼,发现输入都好少,而且看起来没有图论和数据结构题,啊? 看大屏幕系统终端发现有同学已经同步了四题的 cpp,于是我也先把样例复制到考试文件夹,然后把四题的源代码文件和文操先弄好了。 赛时 开考前一分钟发现已经发卷了,遂打开看题面。 啊?七选五…我开的是英语卷子? 看了一会发现这不就是错排列 plus,应该可以一样的去推,遂推柿子。桌面空间有点小,草稿纸写起来不舒服,以及只给了一个小样例,离谱。 按错排列的思路推了半天发现行不通,因为递推到 $n+1$ 的时候不知道对应正确答案是否被前面选过,好像加一个状态也不行,加限制也不太行。 没什么思路,跑去看一眼其他题…发现都好难的样子,然后火速回来。推了一个 $O(n^2)$ 的式子,感觉没问题,把里面的组合数展开然后消了一下阶乘,发现如果换一个主元的话就可以把分母提出来,这样只需要算一次逆元,遂推推推。 1h 过去,推到后面想出了一个基于错排列的 $O(n)$ 式子,刚写一部分就发现不对,然后再回去看原来的式子发现也不对。遂难绷,1.5h 打完暴力,测了几个错排列的样例没问题就过了。 开 B,一开始感觉要拆位后面发现不太对。想了一会之后认定操作二是没用的,因为或运算只会把 $0$ 变成 $1$,而与运算需要的是 $0$,异或运算需要的是相同。首先显然可以异或自己,也可以把数字的任意 $0$ 取出变成 $1$ 其他变成 $0$,这样一做与运算也可以一次性变成零。然后想错了,误以为先异或任意数字再与上同一个数字一定会清零,所以发现只需要找到每个函数的最小值或零点去尝试即可。所以答案只可能是 $\min{x,f_i(x),k,f_i(k),2min,2f_i(min)}$,读入每个函数的时候检验一下就好了。 后来发现不对,先异或再与也不一定会变成零。但是发现答案不超过 $2$,因为要么直接与 $1$ 可以变成 $0$,要么先异或再与也可以变成 $0$,因此答案最大为 $2$,所以只需要考虑一下什么情况可能会取到更小的答案就好了。 如果函数 $\Delta\ge0$,那么最小值才有可能可以取到 $0$,否则由于 $a,b,c$ 都是整数,所以该函数的每个整数坐标点都是经过格点的,那么就最低会取到 $1$。 所以这里继续猜测,答案只会从取最小值和取 $1$ 中选?于是火速开写,过掉了所有样例,但是没有任何证明,也没有对拍。但是只有十多分钟了,后面的暴力有些紧张。 火速看完 C 然后开码暴力,谢谢你 next permutation。 最后五分钟开 D,读题好像是 KMP,坏了,暴力都写不出来。虽然可以暴力真的拼好之后用 find 计数,不过来不及了,遂输出样例跑路。 最后一分钟检查了一下四份代码,遗憾退场。 赛后 发现大家都考的挺炸裂,不过高一成绩都比我想的要好吗,感觉他们其实剩下打满暴力的话分应该比我高。 赛后在校门口找了一家沙县小吃吃了午饭,然后想了一下发现 D 是不是跑两次 KMP 就可以了。然而给的考试范围里面没提会有字符串高级算法,所以没复习 KMP,虽然原理很理解,但细节不太熟悉,感觉就算时间充裕也没办法裸写出来,就是有点可惜暴力分,无所谓了。 ABC 出场后也没有思路,C 的话具体应该要看给的那个函数来贪心或者 dp? 洛谷的大凶说的确实没错,可恶。 后日谈 2024/06/16:哈哈。 2024/07/07:120 分,喜提 2=。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/6/16
articleCard.readMore

学习笔记 | bilibili 远古弹幕查看

起因 最近想要在某个视频查找一些 2020 年的弹幕,发现在 2024 的当今通过官方弹幕列表查询最早只能到 2021 年的了。 然而测试了一下拉取弹幕数据的 api,2021 年以前的弹幕都还有保存,也可以拉取到数据,因此只需要前端绕开时间限制即可。 方法 在 b 站对应视频页面点击”展开弹幕列表“,点击下方的”查看历史弹幕“,在日历中翻到一个亮着的可以点的日期。 接着 F12 打开开发者工具,找到那个日期对应的按钮的 html 代码,其 data-timestamp 这个属性里的值就是对应的时间戳(单位为秒),换成自己要查询的时间戳,然后点击修改后的日期即可。 附录 参考文献 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/6/7
articleCard.readMore

学习笔记 | 旧版本 QQ 聊天记录导出

起因与结果 闲来无事,注意到了七八年前使用过的旧手机,希望查看上面曾经的 QQ 聊天记录。 然而手机十分老,QQ 版本也很老。手机上的 QQ 因版本过久无法打开,但更新到最新版也因为手机版本过低无法安装。总而言之,这个手机不可能打得开 QQ 了。 网上聊天记录导出方法有许多种,然而我这旧手机不能 ROOT,自带的备份软件备份不了 QQ,会提示权限错误。捣鼓了半天没有可行方法,可谓是山重水复疑无路。 所幸找到了这篇文章,其中提到的第四种方法,尝试过后成功导出记录。 但有必要事先说明:不清楚是我的设备的问题还是方法本身的问题,总之虽然成功导出但只有一部分,并没有达成符合预期的效果,但至少看到了以前的部分聊天记录还是颇为欣慰,毕竟也只剩下这一个办法了。 当然其实后面又想到了别的方法,可能会有更好的效果,后文会提到。 我的情况 手机 Android 版本为 4.3,内存空间寥寥无几,系统过旧无法安装新版 QQ,未解锁 OEM,无法 ROOT,可以开启开发者模式使用 adb 调试。手机上安装有远古版本 QQ,也无法使用。手机自带备份软件,但无法备份 QQ,会显示权限错误。 导出方法 工具 由于资源比较难找,这里我将导出过程中我所使用的工具打包分享,均来自于网络,不一定适合所有情况。 » QQ_recovery.zip « 原理 旧版(非 QQNT 版本)QQ 的聊天记录相关文件主要存储于 /data/data/com.tencent.mobileqq/ 下的 databases/<QQ号>.db,databases/slowtable_<QQ号>.db 以及 files/kc,只需要获取到这三个文件我们就可以导出基本的聊天消息文字记录,更多的图片、语音等也在该目录下可导出。 获取该文件一般需要 ROOT,因此这里我们只能使用别的方式,我们可以利用 adb 的备份功能将该文件提出出来。 使用 adb 备份 首先手机开启开发者模式,允许 usb 调试,然后连接 adb。 在 AndroidManifest.xml 中有一个参数为 allowBackup,表示该软件是否允许使用 adb 备份。尽管我手机上已经是七八年前的版本,但该属性已设置为 false,但好在发现 QQ 更远古的版本中该属性还是 true,因此这里我们需要保留数据卸载并安装允许备份的版本。 保留数据卸载: adb uninstall -k com.tencent.mobileqq 安装允许备份的旧版后,进行备份: adb backup com.tencent.mobileqq 备份时需要在手机上点击同意,备份可能需要一些时间,会在当前目录下生成一个扩展名为 .ab 的备份文件。 如果备份文件只有 1kb 那就是失败了,可能是软件版本不对或 adb 与设备 Android 版本不对应。旧版 Android 不兼容高版本 adb,我这里 4.3 使用 1.0.32 版本的 adb 是可以的。 提取数据 提取出数据后,我们这里使用 android-backup-extractor 从.ab 中提取文件: java -jar abe.jar unpack backup.ab backup.tar 之后直接解压即可。 导出聊天记录 这里使用的是 QQ-History-Backup 这个库。 在导出的文件夹里找出我们需要的文件之后,放入对应的文件夹内,接着按照教程使用该工具导出即可。 另一种或可行方法 在备份完之后,.ab 文件其实也可以使用 adb 将其恢复到其他设备上(例如模拟器)。 我们可以选择在新设备上将 QQ 更新到最新版尝试打开,也可以从可以 root 的新设备上更方便的直接拿到我们需要的文件。 这两种方法我都没有试过,或许效果比我更好。 附录 参考文献 安卓QQ聊天记录导出、备份完全攻略 - roadwide - 博客园 GitHub - QQBackup/QQ-History-Backup: 【停更】QQ/TIM 聊天记录导出为 HTML,支持图片、语音,可 GUI 与 非 GUI 操作 (Python) GitHub - Yiyiyimu/QQ-History-Backup: QQ聊天记录备份导出,支持无密钥导出,图片导出。无需编译有GUI界面。Backup Chating History of Instant Messaging QQ. GitHub - roadwide/qqmessageoutput: 安卓QQ聊天记录导出 GitHub - 117503445/qq_get_message: 2020年从安卓QQ数据库提取聊天记录 利用adb备份app的数据 - 小七闲 - 博客园 ADB读取和备份安卓应用数据(无Root)_adb backup-CSDN博客 GitHub - nelenkov/android-backup-extractor: Android backup extractor 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/6/7
articleCard.readMore

拾光纪行 | 省选联考 2024 游记

前言 第二次省选了,这次的心态又与去年有所不同。 一来是因为,NOIP 成绩并不优秀,不抱希望,也就没有压力。二来是因为,很久没有碰 OI 了,自 THUPC 后还是会来机房,但是来机房基本只是找机会跟女朋友聊天和看着和她的聊天记录发呆,顺便补补文化课。 毕竟实力就在这个地方了。如果我 NOIP 后继续停课到现在,进队的概率或许还能看得见,但我最好最好也就是打个 Ag 潦草收场,然后去跟本来就能裸分的大佬抢清北羟基。什么啊。 去年参加省选时也没多大希望,高一嘛来试试水,本来想着还是高二冲队,谁知道 NOIP 后半场一败涂地直接退役。 但不管怎么说,这次比赛也宣告了我本赛季的结束,姑且是写了游记。下一次重新面对 OI,应该是年底了,届时是真的退役赛了。 另外,其实去年参加时也写了游记,但并未公开,本次一并附在附录中。 游记 day -4~-1 考前一周才想起来三月初要打省选。 谁爱打谁打。 什么,还要占我一个周末?去你的吧。 什么,可以提前一天回家?我马上来。 本来打算 day0 早上回家,但是 day-1 是我可爱的女朋友的生日,所以特意提前到 day-1 晚上就回去了,码了好久的字然后定时发到空间了嘿嘿。 当然这一周也不是什么事都没干。day-4 vp 了 YDOI R1,打了 220 跑路了。day-3 vp 了 HBOI day1,赛时 T1 死活不知道哪里错了,只得了 50 pts,而且对的还是后半部分的点,诧异,day-1 的时候发现是最后扫一遍统计答案的时候忘记判 inf 了,改了之后一发过掉。应该算签到成功了吧。 day 0 上午没学 OI,但是和她断断续续地聊了一些。 中午吃完饭没多久就出发去考点学校了,去年在广州,有些远;今年在东莞,近了许多。先去了酒店放了行李,楼层很高,拍了很多照片。然后去了考点学校报道,领了牌子和赠品。今年怎么有赠品?去年怎么没有?去的时候考点学校的学生还在上课,哈哈。 试机的时候打了个不带懒标记的线段树,测了小数据一次过了,非常好。 之后回酒店点了晚餐,之后开始打模版手速赛。十三道模版题,前十道一路打过去基本都是一次过,最后三道是三种联通分量,忘干净了,边双点双强联通的代码全记混了,打崩溃了,遂跑路。 最后成功比室友多打了一倍的模板,拿下手速赛第一名。赛后看了以前的边双代码,把边双改出来了。点双和强联通一点都不想打,跑路。 本来想着姑且尊重一下比赛要早点睡的,没忍住,和她聊到了一点多。 day 1 早上起来还算精神。 吃了早餐,提前大概半小时赶到了考点学校。打不到车,最后加了钱才打到的。 开考提前三分钟给了解压密码,先看了数据点只看得出来有一题图论。后面发现 pdf 没密码可以直接打开,火速看题。 T1 读题:我题目背景呢?怎么全是公式?好不容易读明白题,想了一会感觉好复杂,后面套回题目背景上之后好理解多了,有了一些头绪。 毫无疑问,先翻转坐标系把目标位置整到第一象限,然后接下来大力讨论,然后写着写着发现讨论不明白。 注意到风吹和移动的顺序不重要,以及具体行动路线也不重要,在某些情况下只关心吹完风之后的位置到目标位置的曼哈顿距离,但存在特殊情况。 自然而然想到,先计算一整轮风吹下来后的情况,枚举最后一次吹风,再讨论出前面轮数,然后总天数取最小值即是答案。 对于一整轮风的情况,贪心的尽可能往目标点走,所以将所有向负方向吹的风尽可能抵消,得到了一轮下来的位移以及可以自由安排的移动距离,接下来只需要枚举后讨论。 写的时候发现讨论不出来,但可以用来 check,遂退一步,枚举后二分出总轮数。 过掉小样例,然后在第二个样例发现寄了;修修改改过掉了第二个样例,然后在大样例发现假了。遂开摆。好想她。 T2 有一些思路,感觉这个博弈是假的,二进制拆位去贪心就好了。但发现这个定向加强很不好做,遂开摆,只打了最低一档暴力和做 $m=0$ 的特殊性质分,特殊性质写了半天 01 Trie 过不去,最后全都上了暴力。好想她。 T3 不可以,总司令。检查了所有代码。好想她。 结束了,四个半小时纯坐牢。出校门的时候风很大,衣服没帽子,耳朵冻没知觉了,打到了车但是要十分钟才到,冷死我了。 缓过来之后突然发现她不对劲,还以为在学校出事了,吓死我了。终于回到酒店,本来想出去吃的,被风冻怕了不敢出门,还好酒店里全天开着暖气,这才活过来。 下午摊着了,迷迷糊糊的和她聊天,等消息的时候好像睡着了一会。中间点了个肠粉当晚餐,之后和她聊了五个多小时,这个周末最美好的一段时间。 day 2 早上六点多起来了一次,发现室友通宵了。又躺回去,好像没睡着,一直到了七点才起床。 和她聊天的中途赶到了考场,提前十五分钟进了考场等开考。 我一点都不想比赛,为什么要比这么久,好想继续和她说话。 开题。 T1 这个博弈有点意思,很显然有一个最基本的贪心。最开始的思路是,从确保 $1$ 不被选开始,尽可能让 $2,3,\cdots$ 也选不上,通过这个找到要唤醒的结点,最后搜一遍出答案。 小样例过掉,第二个样例就寄了。 好想她,我为什么在这里做题。 突然注意到题目一个要求是必须走完所有子树才能回到根节点,所以发现前面思路单纯按数字大小来贪是不对的,应该按当前能走到的最小值来贪。 想到思路了,大改特改。有点分治的想法在里面,维护了结点子树中能够到达的最小结点,然后 solve 会在当前子树中进行贪心,然后会根据决策,进入到不同的子树中去 solve。 写完之后我很满意,debug 了流程很符合我的想法,简直神来之代码。结果还是一样,只过掉了第一个样例。 好想她,我不想做题了。她今天放假,我想找她说话。 手模了第三个样例中的一个数据就发现,问题出在选取要唤醒的结点的地方。试着换了别的贪心方法也过不去,估计这个地方不是贪心而是某种规划。。寄! 我在这里比赛不如拿这个时间和她聊天多好。 最后改成了自己认为正确率比较高的决策方法后就润去写剩下两题暴力了。 T2 发现是我最不擅长的概率与期望之类的题目,虽然我觉得这个概率好像是假的。不管怎么样,我发现当 $n+k\le 10$ 的时候还是可以用 next_permutation 水过去的,于是火速开写。 这暴力都不简单。check 是按照约束对整块碎片建 DAG,然后跑拓扑排序来 check。然后写了个 gcd 和 qpow 完成了有理数取模。中间还记错逆元的次数了,一直写成了 $mod-1$ 次方,我说怎么 qpow 跑出来一直是 $1$。 最后是过掉了范围内的样例,剩下的部分就不可以总司令了,我相信总是会出现一个的( 好想她,为什么不能打完暴力提前交卷。 T3 逆天题面,给我看傻了。后半部分题目不是很懂,索性 puts("1") 了。 还剩下十几分钟,好无聊,好想她。 最后一次完全检查了代码没有出现去年的错误,于是开始玩起了电脑。启动了虚拟机看看,发现 vs code 不会配置,于是在终端用 gcc 编译,结果编译出来的程序无法运行。退出来研究了一下别的 ide。 终于结束了,什么都结束了。耶,终于可以和她说话啦。 比完赛马上就回去吃午饭,吃完之后收拾了东西,酒店退了房,然后就离开东莞回家了。路上因为给她发消息看手机,晕车很严重,差点没撑住。到家之后继续聊天,真好。 后日谈 Day1 蓝黑黑,Day2 紫黑黑,真有你的省选联考。 没想到今年也签到失败了,太菜了,day0 打的模板一个没考到,不过无所谓。感觉最重要的是和她聊了好多,但我觉得还不够,要是假期就好了。 不过接下来还是专心 whk,恋爱脑也得收收。老婆和学校哪个重要我还是掂的清的,不能因为学习耽搁了恋爱(?)。 附录 省选联考 2023 游寄 注:以下是去年写的,格式不好,也没用 Latex。而且当时也不敢提到女朋友的事,就单纯写了游记。现在看着蛮难受的,但总之就是一起放出来了。 Day -1 虽然初三暑假才开始学的算法,但我对自己还是挺有信心的,基本上算法我是不会写错的(但是题目会不会做就是另一码事了),所以一上午在学校机房里就只是简单的看了下各种数据结构的代码后就继续学新东西。看了b站上关于计算几何的一些视频,向量、凸包、旋转卡壳、最近点对、随机增量之类的,除了随机增量以外基本都很简单很好理解。下午就坐了两个半钟的车到省会,先去酒店放好行李然后就去考点了。 结果走错门了,门口一大群家长,然而当时我还没发现,因为看到了好几个学生,估计他们也是打车打错了。好在当时周五下午,等了一会正好考点学校放学,学校大门开了,我就随着那些家长一起进去了。虽然因为走错门所以完全看不懂地图,不过看路标还是找到了报到的地方。因为弱校校队只有我有省选资格,所以教练也没跟着来,于是报到处的老师直接把整个文件袋都给我了,里面除了我的胸牌还有教练的胸牌和文件之类的,打算回学校后拿给他。听到对其他学生说有饭票,考完可以在考点学校的食堂吃,发现自己没有,或许是需要学校提前订吧,感到有些悲哀。 然后就回酒店了,复习主要是看了一下比较容易混淆的一群Tarjan算法还有突然想不起来和spfa有啥差别的dijkstra。然后就学了一下新的东西,比如爬山和模拟退火,还有一些乱搞(骗分但不完全骗分)。 Day 1 昨晚没睡好,悲。但还好比较精神,可能我本来在学校就天天凌晨睡,所以没太在意。早上吃完早餐就直接去考点了,结果走到昨天那个门这时候才发现走错了,不过时间很充裕,后面又坐车坐到了正确的门。在考场门口又看了下昨晚的模拟退火,总感觉能用上。进去后就看注意事项然后等开考,旁边的哥们在趴下休息,我由于第一次参加省选,格外的精神。想着D1能不能切一题暴俩题拿个200。输完神奇的压缩包密码和pdf密码后,一看目录感觉T3名字眼熟,多半是图论。 T1看完之后有点不是很明白,就看了看T2,一眼DCC,但是对于是边双还是点双有点不确定,但是能确定最后缩完应该是棵树。一看对大数取模就知道这题凉了,因为我对于这种树上dp+大方案+容斥很不会,虽然前不久才做过一道,于是就赶紧回T1了。第一眼觉得跟差分有关,但是试了试没推出来结论,所以最后选了线段树维护并查集。写+调一直到了十一点,不过由于已经放弃T2了,所以压力不是很大。最后大样例跑的飞快,写出来的那一刻还是挺高兴的。之后T2再看一眼发现还是不会于是赶紧跳T3,想着剩下差不多两个小时打个暴力四十分,D1就算圆满收官。一看题目,发现挺像之前做过的一道模拟费用流,但是又感觉很不一样。题目看到一半的时候想到刚学的模拟退火,感觉很能做,每次随机让某位员工往下走,结果往后一看发现居然是动态的,心里偷偷竖中指。于是只好敲暴力,结果发现数组开不了那么大,无奈只能放弃66666的点,最后小样例跑的飞快,大样例直接爆数组RE,D1到此结束。 中午吃过饭后就一直待在酒店里看电脑,翻了翻wiki感觉没啥想学的,会的懒得看,不会的看不懂,于是就一直在洛谷讨论区里翻来翻去看大家“关于省选”。然后发现了好多和自己做法一样的,发现T1正解居然只是道贪心,估计我最近数据结构写多了,后来发现确实有差分的做法但很容易挂。然后下午就出代码了,发现我竟然手误给几个没有返回值的方法写了返回类型,寄,T1爆零了估计。没有拿noilinux虚拟机编译一次,不然估计不会有这个问题,确实可惜,但我确实怎么也想不到居然我真的会写出这种低级错误,还是在考场上,幸好我本来就是来试试水,没打算冲队,压力没那么大。 晚上突然在b站刷到了用DQN训练只狼AI的视频,点进去就出不来了。估计因为我学AI的时间是我学OI的时间的好几倍吧,我感觉炼丹才更像是我的主业,OI只是感兴趣?不过前几年就在书的尾页中看到过DQN的介绍,对这一种直接输入屏幕图像的训练方式特别感兴趣,当时还跟朋友说打算用DQN训练某个我很喜欢玩的小众游戏的AI,结果后面找了发现DQN的资料实在太少,而且网上实现的DQN玩的基本都是叫A什么的某种很古早的只有几百个像素的电子游戏,所以最后放弃了,结果这时候突然发现有大佬用DQN实现玩现代游戏,还开源了出了教程,真的很震惊,于是就乐呼呼的看到了晚上都没碰OI。 D1理想分,100+0+48,实际估分0+0+40左右(T1其实洛谷能编译而且打了90,但CCF测评机应该是会CE的),目前民间0+0+30左右。虽然我会说没爆零就是胜利,但实际上我估计会等官方数据出来以后在洛谷交一次T1,再把T1分加回去当做我的非官方成绩,毕竟不是很在意官方排名,只是想看看自己能打成啥样。 Day 2 昨晚睡得还不错,而且这次没走错门。来到考场门口听见有同学说D2肯定有计算几何,我心里偷偷一笑然后赶紧复习计算几何,顺便看了眼模拟退火,然而结果没有考到。进了考场以后隔壁的哥们又在趴下休息,我还是格外的精神。开考念密码,跟昨天的反过来,哈哈。 一看目录,看到game就感觉要来博弈论了,但是我不急,先看T1,发现好像又是大模拟博弈论,寄。当时感觉有点像启发式搜索,但又想不出来怎么写启发函数,考虑他一个网格地图感觉会跟曼哈顿距离有点联系,一看样例又看不懂了,我手推都不知道样例答案怎么来的,想不明白为什么另一颗被困住的红棋也会移动(还没看题解,不过此时写这篇游记的时候我突然有点想法,感觉黑方的启发函数应该是棋子到第一行的距离+两颗红方棋子到黑方的曼哈顿距离, 红方可能是棋子到黑方的曼哈顿距离的相反数?这么一来确实能解决我那个疑惑。显然不是IDA*,那么肯定是A*,并且可以用记忆化搜索标记一下状态来判平局,欸卧槽好像感觉突然会了)。看到输入里面有数据编号就感觉有好事,火速看数据范围,嘶……显然,像我这种考场上没想出的正解的蒟蒻,肯定是针对每一种数据去写暴力啦,看到特殊性质A送分,我就直接跳到T2去了,打算最后一小时再回来写T1。 看到T2,果然是博弈论(至少当时我是真的以为是博弈论)。一上来先输入的时候判断一下能不能满足B互不相同,然后提前计算一部分已经确定的答案然后剔除出去。那么就剩下最后一部分了,都是B有俩选择的,这时候就感觉有点像动态规划,但是想了半天,不会表示状态。好吧,老实暴力。然后,直接敲了个2^n的dfs+回溯和剪枝出答案,测第一个样例过了,测第二个发现直接就寄了。然后死活想不明白为什么暴力都写错了,一直调一直改,差不多到十点的时候我就怀疑是我关于A的部分有问题,于是就转而投靠刚学几天的模拟退火,在A的有的选的部分中随机一个换,然后再对B暴力求答案,结果调半天还是错(现在看来,感觉是前面那一段剔除确定部分的地方出了问题。。),直到快十二点,赶紧看眼T3看看能不能骗分,发现样例好多1而且只对第一问也有分,就写了个输出1和一个随机数,然后回到T1写暴力。发现暴力不会写,只有性质A比较稳,后面的BCD好像都有乱写的成分在的,然后D2到此结束,我的第一次省选也结束了。 看洛谷有大佬评价说D2确实很难,而且T2好像不是博弈论,性质A是匈牙利、C是费用流,但我考场上确实没看出来,而且暴力也写挂了,只能说D2不爆零就是胜利。 D2理想分30+5+0,实际估分20+0+0,目前民间10+0+1(T3没想到骗到了一分hhh) 总的来说,感觉确实很难,但又好像比我想象中的简单(或许是因为我压根不去考虑正解吧),这次官方分数估计就是40左右,至少对我来说已经很不错了,毕竟我才学了不到一年。不过D1T1果然还是略有点遗憾,不过当时忘记随机化了,所以不能指望官方数据全随机,可能确实会T几个点吧;D1T2这类题感觉很难但我可以掌握,再学一年的话应该能做出来,毕竟感觉这一类题都思路很明显,缩成树后dp+容斥嘛;D1T3,明年的我写的暴力估计会更优吧;D2T1出考场才想到思路,D2T2和T3相信明年的我能写出暴力。但值得庆幸的是,虽然我这次犯了很多问题,但我都找到了可以改进和避免的方法,至少下次应该不会错在同一个地方了,而我没有下下次,所以,这次确实攒下了很好的经验,值了! 最终官方成绩没到40,太难看就不说了,不过挺可惜的是后面把D1T1考场代码里手误写的的几个没有return的函数的返回值改成void之后官方数据AC了,然而实际上虽然没有CE和RE但是也没有分。 最后的最后,鼓励一下自己,顺便立个志向:数据删除(注:搬过来的时候这一段我还是删了,实际考的远达不到这个,就不放出来丢人了) 参考文献 我去年写的别的游记。 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/3/3
articleCard.readMore

踩坑记录 | Win11 输入法切换语言快捷键失效

问题描述 因为经常需要输入英文或者关闭输入法,所以一直习惯在 Windows 里添加两种语言,然后用 Alt+Shift 切换语言,就可以很方便的打开和关闭输入法了。 当然,这里的关闭输入法实质上是切换到英文输入法。 不知道我电脑发生了什么(或许是一次自动更新),某一次开机之后我就发现我无法使用 Alt+Shift 快捷键切换语言了,而且右下角托盘的输入法设置里也无法切换成英文输入法、 我的电脑系统是 Windows 11 家庭中文版,其他的版本貌似没有出现过这个问题。 解决方案 最初我发现,打开系统语言设置后将两种语言的顺序调换一下就可以恢复正常。但是开机后会被重置,需要重新进行一遍操作,而且有概率失败,需要重启电脑再试一次。 最开始我也是不厌其烦的这样做了,毕竟不能切换语言更加难受,但最近发现这样的方法失效了,按网上说着试了也不行,最后自己捣鼓出来了一个解决方法: 在系统语言设置中删除英语,然后重新安装。 成功的标志是,在系统托盘的输入法设置中,可以选择 “英语(美国) - 美式键盘” 这一项。 我使用了这个解决方法,目前没有出现问题。 附录 参考文献 网上没一个管用的,无法参考。 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2024/2/16
articleCard.readMore

随笔杂谈 | 2024 新春快乐!

不知不觉间这个博客过了一年多了,迎来了第二个新年,年份归档中也终于能见到第二项。 忙于学业和 OI,这一年里我写和研究的项目越来越少,所学习的事物也越来越少。博客也更新的不多,绝大部分字数还是我的游记,中间网站甚至瘫痪了不知多久。 在这一年里,我也变化了许多,About 中记载的许多内容不再切实甚至不再有必要写下。另外我也从今年开始将按照中文符号使用规范编写文章,博客中还有很多地方使用错误或不准确(如中英文之间未加空格)的地方,会在看到的时候将其修补。 许多想法放弃的同时,也有许多新的想法出现。我会将其记录于 Plans 中,等待未来有机会将其实现。 不知道博客究竟有多少人来看过,但每次收到评论,或是出现新的 backlinks 的时候都会十分欣喜。 值得一提的是,一月初的时候给本博客抢了一个还算不错的萌备号。另外其实年初一的时候因为设备原因写不了文章,但其实文章的时间是可以修改的,是不是当天也无所谓,所以本文章其实是在初七补写的。 这个博客仍然很新,但已经有所成长。期待在新的一年里,变得更新欣欣向“龙”。

2024/2/10
articleCard.readMore

学习笔记 | 为 MuMu 模拟器 12 安装 HttpCanary 证书

update 目前 MuMu 模拟器官方已在 MuMu 模拟器常见问题页面中更新了关于 MuMu12 抓包的官方教程,本文不再适用。 MuMu 模拟器 Charles 抓包教程(官方) MuMu 模拟器 HttpCanary 抓包教程(官方) 原因 最近需要对软件进行抓包,但是 HttpCanary 的证书无法正常安装。按照教程点击安装,但是却弹出”未安装该证书“。这是官方的说法: 部分用户在使用MuMu模拟器X版本的过程中需要进行抓包一类的操作,但发现无法安装导入证书,无法正常抓包,这是因Android7.0之后新版本系统的安全限制导致(Android7.0 之后默认不信任用户添加到系统的CA证书) 官方给出了使用 adb 安装的方法,网络上也有一些相关博文,然而都是使用其他抓包工具,没有使用 HttpCanary 的,所以自己记录一下。 我的 MuMu 模拟器 12 的版本是 V3.6.11(2438)。 由于我过程中没有截图,所以这里只简单用文字描述一下过程,如果有疑惑的话可以查看参考文献中的相同步骤的截图。 安装流程 原理 用 adb 直接把证书移到系统证书位置 /etc/security/cacerts/ 里。 模拟器设置 打开 MuMu 模拟器设置界面。 在“磁盘”中将“磁盘共享”改为“可写系统盘”。 在“其他”中将“开启手机Root权限”选中。 当然,在完成安装流程后这些都是可以改回去的。 导出证书 在 HttpCanary 设置界面中,进入"SSL 证书设置",点击“导出HttpCanary根证书”,选择“System Trusted(.0)”格式,导出证书默认就在 /HttpCanary/cert/ 下,我的文件名是 87bc3517.0。 参考文献里其他的抓包工具只能导出 .pem 格式,需要用 OpenSSL 重新签发为 .0 格式,而 HttpCanary 直接为我们提供了 .0 格式的导出,很方便,减少了很多步骤。 adb 安装证书 如果不了解或者没有 adb,可以先在其他博客中自行学习一下。 首先连接到 MuMu 模拟器并申请权限: adb connect 127.0.0.1:7555 adb root adb shell 其中第二步运行后,模拟器会有一个弹窗,显示“Shell 正在请求超级用户访问权限“,选择“仅此次”,点击允许。 第三步运行后,提示符会发生变化,这时候就进入 Android 的 shell 了。 将证书文件放入系统位置: mv /sdcard/HttpCanary/cert/87bc3517.0 /etc/security/cacerts/ 可以检查一下是否成功移动文件: cd /etc/security/cacerts/ ls 87bc3517.0 确认后证书就安装完毕了,可以输入 exit 按下回车退出 shell。 安装完成 这时候启动 HttpCanary 开始抓包,软件可能仍然会显示提示条,不要紧,打开软件试试,如果能抓到 Https 请求就代表证书安装成功了,可以正常抓包了。 当然,有些软件会有 root 检查,启动会闪退,你可以把 MuMu 模拟器设置修改回去,没有影响。 然而有些软件同时还有证书验证,可能开始抓包后会无法联网(没错,说的就是你,森空岛。),难绷。 后续 写完上面一大堆之后,不知道在写博客的时候是哪里捣鼓坏了,我的抓包又变成之前那样没法抓了,但是当时刚搞完确实是可以抓的。 后面又按照 这篇博客 整了一下,效果也只持续了一下,重启模拟器之后就也不能抓了,不知道是什么鬼。 附录 参考文献 网易mumu模拟器安装下载charles的CA证书 - 悟透 - 博客园 MuMu模拟器X版本如何安装证书?_MuMu模拟器_安卓模拟器 burp抓包mumu模拟器_mumu模拟器如何安装信任凭证-CSDN博客 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/12/31
articleCard.readMore

学习笔记 | ripgrep(rg) 如何排除文件夹

前言 最近又开始捣鼓博客模板,要搜索关键词的时候总是被 public 目录下的文件干扰,但直接删除这个文件夹的话又会导致 git 需要重建,所以就学习了下搜索时排除文件夹。 命令 rg "content" -g !public 这样就可以在搜索 “content” 的时候排除 public 文件夹下的文件了。 附录 参考文献 ripgrep(rg)忽略某些文件(exclude, ignore) - Tokubara - 博客园 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/12/10
articleCard.readMore

踩坑记录 | OneIndex 首页出现 HTTP ERROR 500

踩坑 背景 最近突然发现自己的网盘首页上不去了,然而其他页面却能打开,唯独首页不行,怪异。 出现问题前后没动过源码,onedrive 账号当然也没有问题,查了 api 手册貌似也没有变动,重启服务器也是一样的情况,怎么回事呢。 发现主机服务商帮我自动升级了 PHP 版本,猜想应该是由于版本升级带来的兼容性问题。 问题描述 无法打开 OneIndex 的首页(根目录),但其他页面可以正常访问,尝试过重启甚至重装都是一样的情况。 这里我的 OneIndex 是自己以及别的大佬魔改后的 OneIndexN,但从代码上分析,应该所有的未经修复的 OneIndex 版本都会出现该问题。 解决方法 修改 /lib/onedrive.php 的 urlencode 方法: static function urlencode($path){ if($path == '/') return ""; // 在最开始加入这一行 foreach(explode('/', $path) as $k=>$v){ if(empty(!$v)){ $paths[] = rawurlencode($v); } } return @join('/',$paths); } 以 GitHub - xieqifei/OneindexN 为例,这一行代码应被加在此处。 问题应当被成功解决,现在应该可以正常访问首页(根目录)了。 分析原因 经过排查,应该是由于 PHP 版本升级(我这里是被升级到了 8.2),/lib/onedrive.php 中的 urlencode 方法出现了问题。 这个方法的作用应该是将 $path 分割后重新拼接,但对于根目录的情况,由于上下文传入的 $path 仅为一个斜杠 '/',处理的过程中出于某种原因而没有起到作用,最终使得 $path 不合法。导致在后续发起请求时传入错误的参数,从而出现异常。因而解决方法就是在开头进行特判。 没有仔细分析,推测是其中的某一行代码所调用的字符串处理方法在新版本中有不同的实现。 当然,其实也有可能是 onedrive 的 api 原本支持上述错误,而后 api 更新时不再支持,这样也会出现此问题,但我认为更可能为前一种情况。 附录 参考文献 列出文件夹的内容 - Microsoft Graph v1.0 | Microsoft Learn GitHub - xieqifei/OneindexN 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/12/9
articleCard.readMore

拾光纪行 | NOIP 备赛停课日记

前日谈 这次 NOIP 太重要了,如果还是打的像 CSP-S 一样烂就真 AFO 了。 确实需要时间去好好复习备赛,遂停课。 10 月 28 日 其实今天白天还在学校,不过经过这一周的多方谈判协调,已经确实准备好停课了。 早上出宿舍时带走了所有衣服,多余的东西全部收起来或者放到柜子里了,到教室后也把桌面和抽屉都清空了。 中午在机房制定了一下接下来的停课计划,大致就是每天早上起来敲几个板子,然后就补之前的网课录播并做做题,有比赛的话尽可能打打。 由于我自己并不是很自律,所以打算量化每日任务,如下: 题目 红 橙 黄 绿 蓝 紫 黑 分数 $10$ $20$ $30$ $50$ $80$ $100$ $150$ 活动 参加比赛 发布题解 完成课程 记录 trick 分数 $50$ $20$ $50$ $5$ 每日要求: 模板题 $100\text{pts}$ 刷题 $200\text{pts}$ 活动 $100\text{pts}$ 其实原本没打算将分数调到那么高,但是想了想,把总分定为了 $400\text{pts}$,暗示我 AK NOIP,这样溢出来的分就会变成 rp 加到 NOIP 里面去( 每周六晚上比赛都蛮多的,错过了 AcWing 的周赛,不过 AcWing 的周赛一直很水,每次参加基本上算是练练签到题的手感。 发现有场 ABC,但是也来晚了,比赛只剩四五十分钟,已经无法报名 rated 了,所以就简单写了下前三题。前两题没什么好说的,C 题一眼切,但我看数据范围总感觉直接暴力好像能过,所以花了几分钟写了暴力试试,发现 T 的很快后还是老老实实离散化双指针了。最后 D 题看完题感觉挺有意思,但剩下五分钟肯定写不完,遂开摆。 然后晚上又打了场 CF Div2,迅速地切了 AB,然后花了一个小时被 C 给切了,看了下 D 也不会,猜个贪心但是已经不想写了。 开始写了一点 CSP-S 的题解。 10 月 29 日 睡到了十点才起床。 打开 CF 发现 rating 变成 1200了,升入绿名! 继续写 CSP-S 题解,写着写着我们省出分了:$80+25+100+0=205\text{pts}$。 感觉我能拿到的分应该在 $100+50+100+20=270\text{pts}$ 左右的,总的来说还是考差了,但貌似也没有那么差。1= 估计没什么问题,听说甚至还过了七级钩线,就是不知道省排多少,唉。 下午打了一场模拟赛,出题人说不难,我信了。 到点开题,发现 A 是一道数数题。仔细分析了一下发现答案是分组后进行错排列,套公式就可以了。 错排列很好预处理,但组合数 $O(n^2)$ 预处理会爆,所以自己糊了个等价的分组公式出来,这样就可以只用阶乘表示答案了,然后顺便预处理一下阶乘的逆元即可。 写完后过掉了样例一,但样例二输出比答案多了一点,手推了一遍发现我的程序的输出没有错,应该是自己思路本身有问题,仔细想了想发现分组后不是错排列而是圆排列! 这下更简单了,随便改改就过掉了第二个样例。然后第三个样例又错了一堆,遂检查,发现我糊的分组公式漏除了一个东西,补回去之后式子就被化简到了更简单的形式,然后把这题过掉了。 开 B 题,询问的拆分方式一眼出,但是不知道怎么快速统计二维前缀和,遂打表。打表出来发现里面存在一些循环节,但找不到什么可用的规律。 打了好几种表,也往二进制上分析了,找不到什么规律,遂写了 $30\text{pts}$ 的部分分的暴力。 开 C 题,不会,但是发现对于水位相同的连续块,其性质是一样的,所以应该会需要合并这些块,遂猜测用珂朵莉树来做。 开 D 题,第一眼以为是莫队。想了很久,口糊了一种字符串哈希+树状数组的离线做法,感觉很正确,但复杂度是 $O(n^2+m\log n)$ 的,貌似能过一半的点,但实现起来超级复杂。 A 题不会写暴力,不知道怎么拍,遂开摆。 模拟赛就这么结束了,ACD 都不会写暴力,B 题只会写暴力,最后以 $100+30+0+0=130\text{pts}$ 收尾。 结束后发现 BCD 都全场没有通过的,甚至 C 题没有人有分,以至于出题人还问我们是不是都没认真打。 A 题我是正解,没有挂分,而且似乎是由于我糊的那个式子,跑的也比其他人快很多,不错。B 题正解是二进制拆位后打表找规律,个人觉得这个思路很好,拆完位后性质也很好,值得记录。C 题确实如我所说需要合并,但不是 ODT 而是并查集。D 题正解很复杂,得用 SA 做 LIS 和 LCS,这个处理方式也很好,值得记录。 总的来说题目很好,但是太难了,考前出题人还说简单,被骗了! 其实下午本来前几天报名了 GenshinOI Round 3 的,这比赛我参加过 R1,R2 也报名了但未参加,结果没想到 R3 正好撞上模拟赛所以也报名了未参加。 晚上熬夜写完了 CSP-S 题解,写的很全,四题一共八种做法全写上去了。 第一周 10 月 30 日 睡到了快十点才起床。今天开始算是正式停课了。 起床后先按计划打了点板子。其实我一开始决定每天早上打随机几道板子,是因为 exlg 有题单内随机跳题的功能,但是今天早上才发现原来那个随机跳题只能跳到我没做过的题目,所以不能用于随板子题了,于是就变成了闭眼随便抽。 最后是随到了 LCA、割点、笛卡尔树的板子,挺快敲完了,模板 $+160\text{pts}$。然后就开始补这几天的停课日记。 没错,因为前天回来之后先写了题解,一直没什么空闲,所以今天才开始写停课日记,前面的几天都是刚刚补的。 十一点半的时候听说教练忘记报名 NOIP 了,距离报名截止还剩半小时,急急急急急!最后几分钟把报名信息交上去了,好险,停课第一天就差点被迫 AFO 了。 下午我的新键盘到了,是我今年的生日礼物,双十一打折入的牧马人 K200!之前用的基本都是附赠的几十块钱不到的键盘或者笔记本自带薄膜键盘,第一次用这么好的,打起字来感觉都不一样了,敲代码舒服到飞起,状态极佳! 晚上在学分治,P7482 敲到一半去打了场 CF Div2。 A 题很简单,由于 $n\le20$,直接特判一下需要修改的位置是否是 $1,2,4,8,16$ 即可,读完题就切掉了。B 题思考了十几分钟后也会了,直接 $O(30(q+n))$ 跳就好,于是 B 题也过掉了。C 题好像有一点点眼熟,摸了下样例感觉很能贪,于是直接写了个贪心,交上去结果直接过掉了,尽管我一点都不会证。D 题不难,但是我二分打了个表以为 $g$ 是单调的,结果并非如此,没能过掉这道题。听说 F 很板,但估计我还是太菜了,只看得出来题面长得就很板,但不知道到底是哪种 ds 的板子。 应该算是上分场吧。其实 Div2 能过三题对我来说还算不差,D 题有点可惜,离正解不远了都。 比完之后继续敲 P7482,敲完后发现样例都过不去,检查一遍后发现是忘记做前缀和了,改完之后一发过了这道题。 今天只做了一道紫题,没想到我竟然如此不自律,停课第一天开始就欠分了,考虑之后要不要增加个补回来的机制,不然真的 rp 堪忧啊。 模板($160/100$) 刷题($100/200$) 活动($50/100$) 今日总分 $310\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $0\text{rp}$。 10 月 31 日 睡到九点多起床。 昨晚 CF 上分场 Rating 加了 126,感觉过几天的 Edu 有望直接升 Specialist 啊! 今天是我的生日,一早起来领了好多红包,开心!停课后才想起来这件事的,发现这次停课还顺便避免了在学校过生日。 发现洛谷英文字体变了,感觉不是很好看啊。 早上起来打模板,这次直接复制所有题号,然后随便找了个在线工具去抽,抽到了字典树、Nim 游戏、逆元 2,难度黄绿绿。 发现确实不是很熟,字典树虽然打的很熟但看漏了 $|\Sigma|$ 的值域,Nim 游戏直接把必胜必败条件记反了,逆元 2 的 exgcd 还写挂了。 看来多些模板还是有很大好处的! 上午发现昨晚写的题目还能投稿题解,于是写了一篇投上去,不知道能不能过。 中午开始想 P6406 这道题,想了一会只会一点外层做法就去看题解了,看了一会突然好奇写题解的大佬是怎样的人,于是点开他博客看了好久游记和回忆录,突然发现我好像很久之前就看过他的博客,而且至今都不知道他 NOI 打银之后到底签了哪个大学…… 两点的时候发布的题解就过审了,这么快的嘛! 听 kkksc03 说据说大部分省的省选要提前一个月,千万别啊! 一个下午半颓不颓的对着题解码码码改改改,好不容易写完结果答案差了好多,对着题解调了好久最后发现是少预处理一个前缀和,总算是过了这道题。 突然发现我对自己的要求貌似只有一天两道紫题,但是我写的挺痛苦的,一时间不知道自己到底颓还是没颓。 可能是题目问题吧,分治的合并写的太痛苦了。那道题其实还是有很多很好的 trick 的,但我也完全没有写题解的欲望了。 写完那题之后刷到了 LibreOJ 的一个帖子,于是上洛谷看了看他的比赛,发现正好还剩几小时结束,想着放松一下就打一打,结果 T1 一点都不普及-,感觉都快能评普及+了,想了十几分钟才会,也是挺好的一道题目!后面两题就比较简单了,T2 推推式子就行,T3 就是 SCC 板子,愉快 AK 了。 小比赛也是比赛!活动得分 $+50\text{pts}$。滑稽。 请教了大佬后把昨晚的 F 题过掉了,确实板,不过逆序处理操作我上次见还是在 LCT 里见到的,把删边转为加边。 然后去做了昨晚 D 题,死活过不去,一直 TLE,后面删了一个取模就过了,看来 __int128 的取模真的是慢的离谱。 最后做了道绿色的单调栈练习题,由于双倍经验,总算是凑够了 $200$ 分。我给自己的要求到底算高还是算低? 家长联系到了一直很想去的集训的学校!算是个好消息吧,尤其是今天还是我的生日!高兴了很多,不过明天一早就得去,感觉家长和教练比我还急……不想早起。 去那边的话只能带笔记本去了,所以今晚还得提前给笔记本准备一下各种软件和文件之类的,打算到那边继续写日记,就是可惜昨天才到的键盘马上就用不到了。 模板(130/100) 刷题(200/200) 活动(70/100) 今日总分 $330\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $0\text{rp}$。 11 月 1 日 被迫七点起床。 一大早出发了,坐了两个多小时车到了集训的学校,上次来还是省选前。 一来老师先让我做了套他们的 CSP 模拟题,这应该也算比赛吧? A 题求树上每个点的经过这个点的链的最大点权和,看完题就会了,写了几分钟过掉了样例,结果被自己造的小样例 Hack 了,发现这个链也可以是在某个祖先处往下折,然后马上又会了,感觉有点像之前做过的题,直接维护以子节点为起点的前两个最大点权的链,然后再跑第二遍 dfs 统计答案即可,然后一发过掉了这道题。 B 题推公式推了半天,写出来的代码却不对,诧异。遂看机房其他大佬的代码,发现别人的公式跟我的公式有一个符号是相反的,但是我自己推了感觉我是对的,然后再按自己的公式改完发现还是有问题,诧异。 C 题要求统计以每个点为右端点的满足条件的区间个数,第一眼就想到了分治,但是只会做最大值在右边的情况,如果最大值在左边的话不知道如何为右端点统计答案。遂看机房其他大佬的代码,发现对方统计答案使用了差分,啊?想了一会,注意到题目条件具有单调性,所以可以枚举 $l$ 之后二分出一个满足条件的最大的右端点 $r$,于是 $l$ 就对所有 $r’\in[mid+1,r]$ 都有 $1$ 的贡献,需要区间加,于是上个差分,最后做一遍前缀和就把答案统计好了,妙! D 题一看就不会,过。 剩下的时间把今天的模板随了,虽然人在外面但还是得继续刷模板,然后今天随到的是最小生成树、最长公共子序列、欧拉路径。其实我的规定不是每天三道而是每天打满 $100\text{pts}$,但不知为何总是刚刚好要三题才能打满。 然后做了几道 NOIP 计划的作业,一道红一道橙,橙题是前年 J 组 T2,我第一眼甚至以为要上平衡树,后来发现值域 $600$,直接桶排即可。 晚上继续补洛谷课程,学单调队列的时候突然学到了最大子段和原来可以维护最小前缀和来做,我去,还能这样?之前做还是用的 $dp$ 讨论每个点选还是不选,离谱。想到了求最长不降序列这个问题,刚学的时候还是敲的暴力,前一段时间突然想到了树状数组的做法,很感慨以前的简单题居然还有更简单的做法! 顺便还问了一下今天上午的 B 题的公式以及枚举范围,Alex_Wei 老师说我推的公式没问题,应该是我写挂了,我也这么觉得,然后还帮我简单证明了枚举范围,非常厉害! 晚上机房同学发来一道题,做了一晚上发现我也不会,问了认识的大佬说肯定是区间 dp 但还是不可做,后面查到全网唯一题解结果只有四行话…… 模板(90/100) 刷题(50/200) 活动(50/100) 今日总分 $190\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $0\text{rp}$。 11 月 2 日 停课,但是还是得七点起床上学。 上午有一场模拟赛,听说难度巨高,遂开摆。 A 题给了一棵树,每个点有一个分数,可能为负。初始时得分为 $0$,第一次经过一个结点时可以获得该点的分数,可以多次经过同一个节点,问是否存在移动方案可以从 $1$ 走到给定的目标结点 $t$ 使得过程中保持得分非负。 一开始打了个搜索,找到 $1$ 到 $t$ 的那条链,对于不被链经过的所有子树都做一个压缩,得到进入这个子树后最后最多能得到的得分,然后链上跑一遍,有分就收集,看过程中得分是不是始终为正就行了。 敲起来很快,样例测起来很错。对着好几个 $n=15$ 的样例调了半天过掉了,然后这时候只差最后一个小样例还没过了,把树画出来之后发现卧槽我做法假了,因为我压缩子树的时候没有考虑到过程中得分会不会小于零,这样的话进入这个子树时的得分不同,能得到的分数也不同。 看看 B 题,好家伙,数论,对着一个看不懂的柿子求 lcm。看看 C 题,好家伙,暴力都写不出来一点! 遂开摆,11 月份第一场模拟赛喜提爆零。 赛后讲了题,第一题确实是这么个理,所以对每个节点都维护一个可重二元组集合 $(i,j)$ 表示得分大于 $i$ 时进去能额外得到 $j$ 的得分,将子节点的全部二元组加入父节点后贪心地合并,然后最后再在 $1$ 到 $t$ 的链上跑一遍贪心地得分即可。 而 B 题可以用等比数列求和公式转换一下,然后用 min-max 容斥转成求一坨奇怪的 gcd,当然,我不会。C 题更是神秘网络流,最大流转最小割的做法很神奇。 今天洛谷运势为大凶,我试过了,是真的,今天随模板题,随出来LCA、Dijistra、SPFA判负环,三道黄题总计 $90$ 分,还差十分所以不得不再多随一题,结果随出来一道 CRT!再看了几眼大纲确认 NOIP 真的会考 CRT 之后不情不愿的打了,于是今天就打了四道板子,三黄一蓝总计 $170$ 分。 把昨天说写挂了的 B 题重新写了一遍,还是一样的连样例的过不去。调了半天发现原来是区间左边界应该对枚举值取 max,否则多统计了一些答案。于是顺着这个 max 去推公式,终于发现了错误原因,并不是我写挂了,而是当枚举到一定大小之后会导致左端点变成负数,所以要作处理,顺便按同样的做法推了右端点的关系式,成功推到了昨天问 Alex_Wei 老师的那个枚举范围的那个公式,感觉一切都合理起来了,我真强! 下午写了一道 P8945,刚开始写的迷迷糊糊的,后来受不了了于是索性重写,直接不想这么多先 $O(n)$ 一遍把那些鬼前缀全部处理好,然后再重新跑一遍来求答案,总算是理清楚了,一次性把题过掉了。 但是跑的奇慢,于是就开始压缩,把两次遍历合并,这一次就清晰多了,顺便上了个快读,直接排到了最优解第五,随开始卡常抢最优解。 换成关闭同步流的 cin 试了下发现时间慢了一倍,诧异。然后用从大佬那里贺来的 fread 成功抢到了最优解。 下午放学前和机房大佬 vp 了一场 Edu,切了 A 题之后成功被 B 题切了,偷偷看了眼洛谷题解,然后去做 C 题,半天做不出来,后面发现做法假了。最后才知道原来这场 Edu 难得离谱,C 题都 1800 了,这真的是 Edu?? 晚上继续补洛谷网校,终于看完第一节课了!突然发现讲倍增的时候讲到一题博弈论+倍增,我好久之前还随到过这一题!当时盯了好一会连怎么博弈都不知道,现在上课的时候讲到这道题目读完题就看出来 Nim 了,有些感慨。 模板(170/100) 刷题(80/200) 活动(50/100) 今日总分 $300\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $0\text{rp}$。 11 月 3 日 洛谷运势:凶。是真的。今天模板题写的 ST 表和树状数组 1,为什么只有两道呢?因为接下来我随到了 exgcd 和平衡树,然后我说服了自己不写这两道。 上午 vp 了一场 edu,我 C 题罚了三次时,最后只过了三题,一起的大佬已经切到 F 了,其实感觉 D 题会了,但是最后没 rush 出来,看来实力还是太菜了。 vp 完之后继续看昨天的 min-max 容斥,今天看懂了!然后要了一份代码读明白了! 中午和下午就继续上洛谷网校,进度快了很多,估计是因为第二节课简单很多吧!而且还做了一道灰题,具体难度其实我觉得有蓝到紫,打算有空可以写个题解! 放学后突然自己想出来一个 min-max 的简单证明,感觉理解的更深刻了! 晚上打了场 cf,A 题切了,B 题勉强切了,C 题想了一会就会了,但是一直调不出来,好不容易过了所有样例,一交发现 WA on 1,怎么本地测的和 CF 上跑的不一样呢,怎么会是呢?后来发现数组开小了,循环的边界处理也有问题,改了四五十分钟才过掉。后面就一直在看 D 了,感觉有规律,打表半天一点规律都没瞪出来比赛就结束了。 考后阅读到了一篇代码,发现 D 题应该从拆位上面去考虑!其实之前有场模拟赛的某一题就考到过拆位,对每一位进行考察,下次看到异或应该主动往这方面想,深刻。 本来今天模板只有 $80$ 分,但睡前看到总分 $390$,能完成任务的机会不多,为了攒 rp 决定拼一把,又随了两道 Floyd 和单调栈,成功拿到 $440$ 的总分。第一次满分,希望溢出的 $40$ 分真的能加到 rp 里面去吧。 模板(110/100) 刷题(230/200) 活动(100/100) 今日总分 $440\text{pts}$,溢出 $40\text{pts}$,挂分 $0\text{pts}$,总计增加 $40\text{rp}$。 11 月 4 日 洛谷运势:大凶。希望洛谷是在帮我攒 rp。 今天周六诶,周六!为什么这边的学校周六都不放假的啊!! 上午打了一场模拟赛,比前几天简单了许多,但还是爆难。A 题构造,不会,遂打表。找到一个规律,感觉贪的很对,统计答案也很好做,写了写把全部样例过掉了就交了!B 题推了一下之后写了个分治,感觉很对,但是样例太少了,遂写拍子,一拍就挂!改了之后再拍,继续挂!再改,再拍,哦豁,我做法假了。C 题怎么看怎么不会,D 题怎么是概率啊再见! 打完之后发现 A 题挂到了 $60\text{pts}$,不知道是贪心假了还是哪里写的不对,题解写的很抽象。B 题做法确实是假的,正解是经过一些抽象操作后转化成二维偏序。 下午一边继续补网课一边写题,中途还帮学校一位高一的后辈调了道题!写了一道 P5522,状态压缩后用线段树维护,很不错的一道题!另外还打了两道线段树求区间最大子段和的板题,当初第一次看到这个做法还是在某个初赛阅读题里面。然后突然扶苏姐姐就开始讲矩阵转移了,也就是名为动态dp(DDP)的神奇做法!现在已经会四种求最大子段和的方法了(虽然矩阵转移本质就是 dp)。 发现昨天那场 CF rating 结算了,这次没有通知我。加了九十分,上 specialist 了!! 网课又上完一节,感觉现在进度快多了,不知道是因为之前比较难还是之前我太颓了。学到了很多新的结论,比如说如果 $a\gt b$,那么 $a\text{ mod } b\lt\frac a2$,这个结论可以启示我们如果每轮取模,其实总共是 $\log n$ 轮的,所以一些看起来暴力的做法的复杂度其实是正确的。以及关于 lxl 的"线段树其实就是特殊的 01Trie"这一说法也很值得思考。 晚上打了场 ABC。A 题切了,B 题切了,C 题也切了。D 题稍微想了想,发现好像给的条件其实相当于连边后判二分图,直接染色就好了,第一次交上去一半 WA 了,后面发现是 dfs 时传错节点编号了,这都能过一半的点?第二次交上去还是 WA 了,然后突然想到图不一定联通,应该对所有联通块都判二分图,马上改好交上去了,过了,吃了两发罚时。 这次打的好快啊,二十多分钟就过掉 D 了,抓住机会上大分!遂开 E,给出了一个计算 rating 的神秘式子。观察式子,发现选取的比赛越多越好,其次越往后的比赛的得分越大越好,猜了下其实在求最长不降序列,写完后交了下发现假了。继续观察式子,发现如果倒序处理,那么指数就跟总数无关而只跟已选个数无关了,那岂不是直接 $O(n^2)$ 倒序 dp 就可以了?迅速开写,写完之后样例全过,自己也感觉很合理,结果交上去 WA 了一半,不知道是怎么假的。 发现很多人写了 F 题,遂开 F,读完题目发现这不就是 P1502 吗?我居然能在 ABC 做到原题?于是直接贺了自己的代码过来,改了改交上去,结果发现有一个点始终过不去,啊? 感觉代码一点问题都没有,样例全都过了,完全无从下手,不知道怎么调,于是 ABC 就这么结束了。结束后突然想到可以贺别人题解过来交,于是随便贺了一篇改了一下,一发就过了。 啊?合着原来是我自己的扫描线板子代码本来就有问题,最丢人的一集。赛时没想到题解,不然多过一题应该能多上好多分! 结束之后马上就更新 rating 了,加了一百五十多分,上棕名了! 晚上仔细想了想,虽然很不想写 splay 和 exgcd,但毕竟 NOIP 还是有可能考的,所以今天没遂模拟题,而是把这两道昨天随到的补了。写 splay 的时候有点手足无措,感觉不会写,但是记住了扶苏说的“splay 操作其实就是模拟“,莫名其妙地写出来了,基本没什么大问题。 今天分也挺高的,都是为了攒 rp! 模板(130/100) 刷题(240/200) 活动(150/100) 今日总分 $520\text{pts}$,溢出 $120\text{pts}$,挂分 $40\text{pts}$,总计增加 $200\text{rp}$。 11 月 5 日 周日了,九点多醒的,睡够了。 今天不光颓,还唐。洛谷运势:中平,总算不是凶和大凶了。 醒来之后没有写题,在 OI Wiki 上学了下矩阵,发现这个东西其实不难,然后颓到了中午。 中午考了场 AMC。没错,是 AMC(美国数学评测),不是 ACM。其实上面的数学题大部分不是很难,由于我没草稿纸,所以都是在手机备忘录里打的草稿,偶尔用下计算器,有一个数列问题还查了 oeis,最后比出来一百多分,貌似按去年分数线都没奖。 下午打了场模拟赛,是 lxl 出的题。十分钟看了下四题,没看出什么所以然,还以为会是四场数据结构。A 题看起来很贪心,发现貌似直接排序后双指针即可,开写,中间差点把自己绕晕了,最后写完了,过掉了两个样例,结果最后一个大样例错了,啊? 检查了一下代码感觉没问题,遂写拍。结果还没拍上呢,测试数据生成器的时候发现我的暴力写的不对,然后 debug 了一会暴力突然发现这玩意儿有后效性,我写的贪心直接就假了。想了一会想不出正解怎么做,想着既然过掉了好几个样例那应该部分分还是有的,遂开 B。 B 题发现求树上存在绝对众数的联通块个数,但我没学过有关绝对众数的东西啊,顶多做过 Violet 和蒲公英以及 lxl 大爷题。 于是现场开学,学完了摩尔投票法之后发现这事不对啊,树上不能这么做,遂试着搜相关题,结果发现了这道题的序列版 P4026 和区间查询版 P7882,现场看题解学了下做法,有一点头绪但不多。最后还是不知道怎么上树,遂写部分分,链的部分就等于问题回到序列上,由于 $n$ 很小所以直接上个 BIT 即可。然后小数据打算写个暴力一点的 dp,结果发现写不出来一点。 还有一分半结束的时候,听 lxl 说为了模拟真实情况,D 题给我们保留了不可以总司令,遂半分钟内迅速敲了个骗分,直接全部输出 NO。 最后拿了 $70+20+0+19=109\text{pts}$。由于第一题本来就假了,所以也不算挂分,倒是最后一题不可以总司令居然有 $19$ 分,诧异。 考完之后 lxl 马上就开始讲题了,A 题题解看一半就会了,感觉赛时真降智了,这么简单的贪心都看不出来,还写双指针,重新写了下只用十几行就过掉了。B 题正解是优化后的树上背包,我的思路没什么大问题,但由于我不会树上背包所以这题后面一点暴力都写不出来。我确实背包问题做少了,不光树上的,序列上的背包都不是很会做,确实要练练!C 题是个比较复杂的 dp,用线段树进行了很神奇的优化。D 题是从联考搬的题,貌似实际难度不比 C 高。 貌似求满足条件联通块是树上背包的典题,打算明天学学。晚上稍微看了一点下节的课的开头,然后就继续颓了,毕竟是周日嘛。 睡前随了道模板题,又是像之前那种情况,先随了三道黄题:逆元、三分、负环,因为分数不够然后再随了一道,结果是绿题:割点。不过其实还好,除了三分以外都敲得很熟练,看来刷模板战术起效果了! 今天真的是又颓又唐,题没练,比赛也没打好,之前说要写的题解也没写,希望明天开始又能回到前天昨天的状态吧。 模板(140/100) 刷题(0/200) 活动(100/100) 今日总分 $240\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $200\text{rp}$。 第二周 11 月 6 日 周一了,上学,要早起。 上午有场模拟赛,这次没有先看一遍全部题,反正也没机会做后面的题。 开 A,第一眼以为我小学奥数题,后面发现好像不太一样,想不出公式,遂模拟,样例全过,时间貌似还行。写了个拍,不拍答案只算时间,发现会有些点 T 了,尝试优化。想了半天,推出来一个式子,感觉是类似于分块一样先把大块的快速做了,剩下的小块再暴力,结果后面拍起来发现比优化前更慢。但是后面发现虽然拍子上提示我 T 了,但实际单独运行程序的时候又没 T,我也不知道什么情况,所以就挂后台拍着了。 开 B,不会,复制粘贴,诶找到原题了,启动!发现做法很简单,有一部分我已经想到了,看完后花了几分钟写了二三十行就搞定了。其实感觉这题我应该能想出来的,毕竟做法似乎有点显然,不过这次就算积累到经验了!交完后看了一眼 A 题的拍子,已经拍到几千组了,几乎没有超时的,貌似挺稳,但是随机生成的数据感觉很水,有点慌? 开 C,使劲想,发现曼哈顿距离有个绝对值,联想到周日下午打的一场比赛里面拆了绝对值,遂尝试拆,但发现曼哈顿距离绝对值情况有点多,好像不能按那个来。火速打开 OI Wiki 看二分图最大权匹配相关东西,发现里面有费用流做法,想了想发现还是得建 $n^2$ 条边,做不了。最后还是不会,复制粘贴,我去还有原题,启动!原题是 AGC034D,看一眼题解马上就会了。发现确实跟我想的一样要拆绝对值和跑费用流,绝对值拆出来四种情况之后建立四个中心点对应四种不同情况,然后黑点连到源点和中心四个点,白点连到汇点和中心四个点,这样边数就是 $O(n)$,跑费用流即可。火速从我的费用流板子那里贺了段 EK 过来,改一下建图方式就过掉了所有样例,芜湖,我起飞了? 开 D,什么玩意儿?开摆。发现有 $20%$ 的 $n\le 8$ 的部分分,一分钟手动打个 $[1,8]$ 的表,交上去完事。看了一眼 A 的拍子,已经上万了,感觉没问题于是停拍。感觉这次分挺高的,这几天比了那么多场终于舒爽了一回,尽管我 BC 都翻了原题。 考完后出分:$40+100+55+20=215\text{pts}$,啊? A 题看完题解才发现可以推式子做到 $O(1)$,是我蠢了。而且我随机的数据强度太弱,竟然一万组 $1e5$ 的数据都没卡到超时。C 题我的 EK 忘记开十倍边数了,而且数据范围比原题大,费用流只有 $70$ 分,正解要用模拟费用流。 好吧,虽然这样,但这次还是打的不错了。打完比赛后发现洛谷咕值更新了,由于上周交了篇题解所以社区贡献到了 $8$,其实还是蛮好赚的,另外练习情况也增加了几分,不懂这个是怎么增加的,总估值 $311$,排名 $465$。等我下周发到七级钩,然后再刷一周题,再发两篇题解,应该有望提升更多! 中午继续看了一点课,然后就去做上一节课作业单了。先切了一题比较水的黄题,然后由于作业单里有字典树的板子题,联想到我前几次刷板子每次刷到字典树都会写挂,所以打算顺便写一遍。果不其然,又挂了,对着以前的代码调了好几次才过,感觉有必要加强练习啊! 写完之后看了眼大纲,发现卡特兰数我还不会,就去 OI Wiki 上看组合排列相关的东西。 晚上继续补了一点课,然后开始写 P6587,调了一个多小时才过,最后发现是因为几个地方漏开 long long 了,无语,但是这题确实不错,让我体会到了之前说的“线段树是特殊的 01Trie”这一观点确实很有道理。 睡前随模板题,结果又随到普通平衡树了。这次没有犹豫,直接开写,写的非常流畅,最后简单调一下就过掉了,感觉随模板题的做法很正确啊,很有效果。忘了自己下午做了道字典树,又继续随,随了一道 ST 表,几分钟写完然后一发过掉了。 感觉今天状态恢复了许多。接下来计划是 OI Wik 上补一补组合排列、矩阵、各类背包,另外课也继续上,作业单也及时跟进,感觉很有前途啊。 模板(140/100) 刷题(290/200) 活动(50/100) 今日总分 $480\text{pts}$,溢出 $80\text{pts}$,挂分 $75\text{pts}$,总计增加 $355\text{rp}$。 11 月 7 日 起床,再睡一会,起床,再睡一会,起床。 早上打了场模拟赛,但是一上午脑子有点晕晕的。 A 题读完题没什么头绪,只会写 $O(2^n)$ 大力搜索,一看数据范围 $n\le 20$,哦,原来是签到题,直接开搜。里面有一点细节,所以写+调了大概半小时不到过掉了。发现这题没什么能拍的,但这次写的很稳,应该不至于像前两天那样 A 题挂分,遂开下一题。 B 题,基环树上 dp??查了半天跟联通块有关的知识点,发现全都用不上。开摆了,抱着死马当活马医的心态,试着搜了下题目标题,结果这一搜直接搜出来了,果然还是原题!读完题解就会了,原来联通块只是恍的,实际上找个环之后讨论一下可以直接 $O(1)$ 修改和查询。写了半个多小时,几乎没怎么调就一次性过掉了,感觉非常慌,但是我不会写暴力,也没办法对拍。 C 题会写暴力,猜测也有原题,一搜出来后发现正解是比较复杂的分治,后半部分就看不懂了,于是开始自己想做法。跟机房大佬交流了一下,把我的贪心给毙掉了,然后会了他的 $O(n^2)$ 归并排序优化暴力的 $60$ 分的做法。 写完后还剩五分钟,看到这次 D 没法骗分,直接开摆。 出分之后发现自己 $100+0+40+0=140\text{pts}$,怎么会是呢。C 题发现 subtask 绑错了,后面重测之后 $40\to60\text{pst}$,但总分还是只有 $160$ 分。A 题没挂,甚至还跑到了最优解,很好。C 题也没挂分,很好。B 题怎么回事? 后面才知道 B 题中途更新了样例,但我没刷新所以没看到公告。对着新样例调了一会发现我忘记把对反向边也做设置了,加上之后拿到了 $60\text{pts}$,应该是剩下还有哪里写挂了,离谱。 中午继续补课,然后老师讲了 AGC010E,会了,想着刚听完不如写一写,毕竟是道黑题,于是写了二十分钟把这题过掉了,黑题喜加一。 下午补完了第二课的作业的最后一题,总算是 AK 一次作业了!开始写第三课的作业,第一题是 CF1253F,好家伙,一上来就一题紫。听课的时候感觉还行,没想到写起来这么麻烦。类似于货车运输那道题,先建图跑一次 Dijistra,然后重新赋边权后做一次 Kruskal 之后,询问就变成了求最小生成树上两点间简单路径上的边权最大值,倍增求 LCA 的同时维护即可。写了一小时,调了半小时,最后等评测结果特别难熬,一百二十多个点。 晚上回来先打了打板子,第一题裴蜀定理,两分钟打完过了。第二题乘法逆元 2,五分钟写完之后 T 了,稍微改改就过了。第三题非常巧,随到了乘法逆元 1,啊这,一分钟打完过了。怎么今天的板子都是数论? 晚上打了一场 CF Div2。 A 题题面好长,而且出现了我不认识的单词,看了好久才读懂,模拟判定一下就好。 B 题有点难读懂,一开始没理解什么什么叫做 exactly two 和 exist,后面把样例仔细看了看才略微明白一点,貌似也简单模拟一下即可。 总感觉今晚状态不佳,写模拟写的很不顺,感觉没有平时那种码力了,或者说感觉写出来的模拟都好蠢,总之将近一个小时才开 C 题。 C 题想了十分钟,突然发现往回走是唯一的,所以一直往前找看能不能循环即可,五分钟写完交上去结果 WA on 2 了,瞪了一会发现就算不能循环,如果往前推的次数能够达到 $k$ 次也是满足题意了,一分钟改完交上去就过掉了。 最后半个多小时一直在想 D,发现貌似一定存在某种使得 LIS 增加最多不超过 $1$ 的插入方法,感觉将 $b$ 从大到小排个序,如果能直接插在末尾的不升序列中那么一定不会使得 LIS 长度增加,除此之外如果直接插在相等的数字后面也不会增加,以及 $b$ 中所有大于 $\max_{i=1}^n a_i$ 的数倒序插在最前面也不会增加。剩余的既没有在 $a$ 中出现过也无法插在末尾的数就比较难办,可能会使得 LIS 增加,但是同样倒着插在最前面的话可以保证 LIS 增加不超过 $1$。 然后剩下十几分钟马上就开写了,写了个离散化维护数字出现次数,然后用栈来倒着记录答案,一个个插入即可,最后倒序输出答案。写完了,发现样例有一条不对,拉出来单独测试发现又对了,以为是多测没情况,但是检查了好久甚至最后上了 memset 都没救过来,然后比赛就结束了。 比赛结束后 system testing 期间发现我有一处使用了 $a_{i+1}$,也就是默认了 $a_{n+1}=0$,这一部分没清空,所以出现了问题。但是经过交流之后发现我的做法不是最优的,最优的做法也差不多,将 $b$ 从大到小枚举,将 $b_i$ 插入第一个大于等于 $b_i$ 的 $a_i$ 后面即可,仔细想了想发现这样就能保证不会使答案增加了,确实犯蠢了。 感觉这次比赛打的不是很顺,而且题面都比较长比较难读,确实打的不算好。 模板(110/100) 刷题(300/200) 活动(100/100) 今日总分 $510\text{pts}$,溢出 $110\text{pts}$,挂分 $100\text{pts}$,总计增加 $565\text{rp}$。 11 月 8 日 早上起晚了,但也不是很晚。 打了模拟赛。 A 题怎么是期望啊,他建议我先做第三题?所以就去看了第三题。第三题貌似不难,推了将近一个半小时之后 $k=0/1$ 的部分已经没问题了,但是 $k=2$ 的部分讨论起来特别复杂,已经完全不想写了,于是打完 $k=0/1$ 的 $20$ 分就跑路了。 回来看 A 题,发现求期望部分貌似不难,似乎第一个区间只需要区间乘 $2$ 的逆元后区间加,第二个区间只需要区间乘 $2$ 的逆元即可,这不是线段树 2 嘛?火速贺来了线段树 2 的代码,过掉了样例一,结果样例二挂了。稍微推了一会,发现假了,这是全概率事件,不能这么拆,岂可修。 现在只剩下半个多小时了,而我只打出来 $20$ 分,坏了。赶紧看 B,给了一个序列,操作一是整个序列对给的 $x$ 做平均数,操作二求区间第 $k$ 大。题意读懂之后发现不会做,遂写暴力。 但是,反转来了。写暴力刚写完离散化的时候,突然发现操作一不会影响操作二,操作二要求的区间第 $k$ 大就是离散化后的区间第 $k$ 大,变成静态的了,而操作一是对全体 $[1,n]$ 做平均数,那我岂不是在离散化数组上建一棵线段树进行区间加和区间除,然后在离散化后的数组上建主席树求静态区间第 $k$ 大然后回离散化数组的线段树上单点查询就好了?! 火速开写,此时还剩二十分钟,而我直接删掉了写一半的暴力,决定开始 rush 正解。当然,不是现场敲。直接贺来了我的主席树板子代码,调了几分钟就可以用了,然后发现我没写过区间除线段树,遂从网上贺了博客的代码,方便起见直接放到 namespace 里,直接按原代码的用法就可以用!简单改了几处问题之后就可以用了,测了下前几个样例都过了,此时还剩两分钟,赶紧交上去了。最后两分钟前 rush 出了 B 题,心里十分激动。 过了一会就出分了,这次是 $0+100+0+0=100\text{pts}$,啊?我 C 的部分分怎么挂了?检查了一下发现数组开小了,重新交上去就拿到了 $20$ 分。我怎么每次模拟赛总得挂点分? 后面看了题解,发现 A 题期望确实一个一个点看就好了,但是要用排列组合推一个比较复杂的式子,其实我感觉我能推,但我赛时上没有往这方面想。。。至于 B 题,大致上和我的做法差不多,只不过他们只用建一棵主席树求 kth,而操作一则是直接暴力做前面的 $O(\log V)$ 次后整个序列会只剩两种值,剩余的操作一就可以 $O(1)$ 判断了,而实际上区间除线段树里面也差不多是这样的。然后发现全场没有做出来 C 题的,是很复杂的讨论,被出题人骗了,还在 A 题推荐我先做 C,太可恶了。至于 D 题貌似要用 FWT 优化 dp,不会。 中午继续补课,把第三节课也看完了!然后就开始做第三节课作业的第二题,结果写出来之后怎么调都调不出来,调了一个小时后才发现是记错样例答案了,无语。 晚上回来直接开了第四节课,上了状圧 dp 部分内容。然后机房同学发来一道边双缩点的板题让我帮忙调,读了一会之后就调完了,顺便还拿他代码偷偷过掉了这道题。 晚上随了三道模板,第一个随出来的刚好是边双联通分量,好巧!后面两道分别是拓扑排序和单调队列。 感觉今天状态不太对,题目做的有点少,明天没有模拟赛了,希望能恢复过来。 模板(100/100) 刷题(180/200) 活动(100/100) 今日总分 $380\text{pts}$,溢出 $0\text{pts}$,挂分 $20\text{pts}$,总计增加 $585\text{rp}$。 11 月 9 日 洛谷运势:大吉!! 由于今天上午没有模拟赛,所以早上随了三道模板,希望晚上能早点睡。今天随到的是 LCA、裴蜀定理、exgcd,都是我不久前写过的板子,很快就写完并一次过了。 接下来继续完成第三节课的作业,然后发现第一题居然是排水系统。这一道题我去年的 11 月 18 日,也就是去年 NOIP 前一周还做过这道题,当时以为这题只是简单的 toposort 加个 gcd 通分约分就好了,结果只有 $20$ 分。现在重新做了一下,写了个分数类,总计十分钟写完,一次性过掉全部样例,交上去发现只有 $90$ 分,把 long long 全部改成 __in128 之后就过掉了,没想到我先除后乘也会爆 long long,放在当年不能用 __int128 的时候来看这题确实很坑,可能需要用两个 long long 拼个 __int128 才能过。 后面又做了一道 [NOIP2022] 建造军营,去年的 T3,当时我考场上看到这道题直接放弃了,考后对着题解才补了这道题。不知道我补题的时候有没有看懂题解,应该是看懂了,反正之前是过了。现在又重新写一遍,重新看了下题面,很显然的思路是边双缩点后跑树形 dp,设 $dp_{u,0/1}$ 表示以 $u$ 为根节点的子树中有/无军营时的方案数,但后面发现到根节点的边很难讨论,转移方程好像有问题,遂开摆,看题解,发现只需要令 $dp_{u,1}$ 里面的方案强制所有军营到根节点的边都要选,再在统计答案时强制 $u$ 到 $fa_u$ 这条边不选即可,这样统计答案也不会重复或者漏,因为 $u$ 到 $fa_u$ 这条边不选时的贡献会在 $u$ 处被统计,选时的贡献会被合并入 $fa_u$ 后统计。写完之后一次性过掉了前三个样例,第四个死循环了,后来发现是栈开小了,开大之后就过掉了,交上去喜得 AC。 然后就去写了一下 P9565 的题解,很有意思的一道题目,二进制分析后用 $62$ 个并查集维护 $62$ 个子图的连通。 中午补了一下课,看了一点树形 dp 的内容,但我发现我基础不太好,背包问题都不熟练,需要抽时间恶补一下! 下午打了场模拟赛,开题不会 A,感觉很树状数组,很容斥,结果想了半小时都想不出来,遂跳。开 B,啊?怎么是我做过的原题,甚至这题我还写过题解?直接贺了我之前写的代码,一测样例全过直接交。开 C,怎么感觉有点眼熟,等等,我好像也见过?一搜发现,又是原,只不过这道题当时没写出来,后面也没补,又贺了一份题解来好好研究,看了半小时学明白了就交了。回去看 A,又想了十几分钟,突然一拍脑袋拍出来了,直接上了三棵树状数组然后容斥出答案,测了所有样例全过!!天啊,起飞了这场,一个半小时过了 ABC。开 D,不是原,但是感觉很简单,貌似跑两遍 dfs 就可以了?赶紧开写,写完之后过掉了链的样例,但是第一个样例错了一个数,仔细一琢磨发现我做法假了一半,第二遍 dfs 没问题,但是第一遍 dfs 里面转移答案是错的,不能简单的乘起来,貌似得做一个背包。坏了,我刚好背包一点都不熟啊。 考试开始两小时时有事所以提前退场了,等回来的时候比赛只剩十三分钟了,火速回来试着改,但是怎么改都改不好,最后三分钟的时候放弃了,直接回退到第一次写完的版本,寄希望于过掉链的部分分,然后开始摆烂。 过了一会就出分了,$90+100+100+0=290\text{pts}$,啊?我 A 怎么又挂分了?看到最后一个点 T 了,检查代码发现,我写了 #define ll __int128,然后改成 long long 就过了,离谱。 总排名 rk5,第一次这么高,也是我 NOIP 前打了十几场模拟赛来分数第二高的一次。感觉这场比赛有点赚又有点亏,赚是因为两道原题,亏是因为少了两个多小时赛时,不然我有信心现场学完背包然后写出 D 题。 晚上听了讲评,貌似我三题做法都和正解略有不同但不多,第四题也确实非常简单。我感觉难度是绿绿蓝蓝,但貌似出题人评的蓝蓝蓝紫,第一次评难度比出题人低,赢! 睡前从机房其他大佬的做题记录那找到一道水紫,做个边双之后直接暴力找一次 LCA 就好了,没怎么调就一次性过掉了。 可能是今天上午没有模拟赛,轻松一点吧。今天状态还是不错的,不过背包确实赶紧安排时间学一下了。 模板(110/100) 刷题(300/200) 活动(120/100) 今日总分 $530\text{pts}$,溢出 $100\text{pts}$,挂分 $10\text{pts}$,总计增加 $715\text{rp}$。 11 月 10 日 洛谷运势:大凶。难道昨天那场二次原模拟赛把我运势耗光了? 今天上午也没有模拟赛,所以先打模板。随到了欧拉路径、ST 表、线性筛。然后就开始写图论作业了。 作业第一道题是一个老朋友:CF786B。我六月初的时候刚学线段树优化建图时写了这道板子题,而且时不时就 UKE,当天调了一个小时都没过。第二天又回来接着调,还是没过,于是就放弃了。过了一个月后,七月初我又回来调了两下这题,还是没过。再过了一个月,八月初我又调了一次,结果还是没过。今天上午又遇到这道题,打算从头开始写,都准备好调一上午了,结果竟然,一次都没调,一口气写完就直接过掉了样例,然后直接交就过掉了!! 估计是得益于,我现在写的线段树规范了很多,并且刷题数量和模板题刷题次数大了很多,可以说是已经形成一种正确且较为优秀的码风了(毕竟我每次重新打模板,都和之前打出来的长得几乎一模一样),所以这题的线段树和 Dijistra 写的特别的稳,因而就一次性过掉了。 后面继续写了几道题,感觉都还行,就是有点困。下午见到了 THU 计算机系的主任,开了一次交流会,大概内容就是讲的 THU 专业和强基之类的事。我很感兴趣,奈何我没这个实力,就不在这白日做梦了。 做了一道 P5344,倍增优化建图,算是挺难的,但是意外的没怎么写挂,但是写完 Dijistra 忘记在 main 里调用了,虚空调试了属于是。后面交上去 WA 了一个点,把数组开大点就变成 RE 了,调了很久,提交了一整页都没调出来,最后直接对着题解的数组大小改才过掉了。 晚上有入门赛,不过我没打。群友问了一道构造题,然后被我切了,我真棒!今天战绩四紫一蓝,感觉还是不错的。明天上下午都有模拟赛,晚上还有一场 ABC,估计会很累,所以今天就早一点睡了! 模板(100/100) 刷题(480/200) 活动(0/100) 今日总分 $580\text{pts}$,溢出 $180\text{pts}$,挂分 $0\text{pts}$,总计增加 $895\text{rp}$。 11 月 11 日 洛谷运势:大吉。今天双十一,并且今天开始是真正的倒计时一周了。 上午是模拟赛,听说是雅礼供的题,很难。 开 A,想了很久,最后只想出来个 $O(n^4)$ 的区间 dp 做法,对比本体数据范围估计这只能叫做暴力了。写了好一会,测出来过了前两个样例,结果第三个样例没过,思考了一会发现这个东西是可以中转的,所以就用类似于 Floyd 的方法预处理了转移代价,然后就过掉了三个样例。虽然最后一个跑了七十秒,但是答案对了。 开 B,不会,遂想暴力,发现可以直接树剖后用树状数组维护,我之前出题出过这个,但是那题造数据的时候才发现暴力更快,所以我想了想就直接上暴力了。 开 C,不会,但是时间很宽松,貌似直接 $O(nmq)$ 模拟都有很高的分。 开 D,交互题,不会。写了个随机数交上去了。 最终得分:$60+20+60+0=140\text{pts}$,一题正解都不会,但是暴力分还是蛮足的。发现全场没有人过了 A,然后 D 也只有一个人过了。 C 题机房大佬讲了一个很有意思的 trick,用 bitset 去判断高维空间的 $n$ 个坐标点有哪几个在一个给定的超立方体内。比如说先看一维的情况,离散化坐标后,我们预处理出 $n$ 个长度为 $n$ 的 bitset,其中第 $i$ 个 bitset 为 $b_i$,所有坐标小于 $i$ 的点在 bitset 内对应的位置设为 $1$。那么一条线段 $[l,r]$ 在一维空间覆盖到的点其实就是 $b_r\oplus b_{l-1}$。对于高维空间,用一样的方法对每个维度进行相同的处理,把每个维度得到的 bitset 全部做与运算,最终得到的 bitset 内为 $1$ 的点满足所有维度坐标都在范围内,也就是在超立方体内了。 至于 D 题,我后面自测数据的时候发现跑的飞快,比答案给定的次数限制基本能快一倍,但是不知道是我写挂了还是评测机动了手脚,本地测都能过,交上去永远过不了,悲催。如果这是正式比赛,我可能已经靠着随机数把这题水过去了。 下午洛谷有场模拟赛。考前听线下的朋友说很难,害怕。 开 A,思考了一会发现质因数分解一下好像就做完了,后面再稍微想了一下发现直接贪就好,往质因数幂次最小的上面加就完事了,但是写起来发现有点麻烦,特别是贪心的操作有点难写,花了半小时才过掉这道题。 开 B,构造题,感觉跟去年喵了个喵有的一拼。对于构造题我一般都不考虑去证明,一般都是先观察样例或者打表找规律,找到规律后再去检验,最后试着证明。结果这题硬是瞪了一个小时都没找到什么有用的规律,遂先跳过。 开 C,优美度之和很好算,但是第 $k$ 小不太会处理。考虑到交换相同数字会带来很大的相等总和的情况,合理怀疑到后面很有可能问到的就是最小的。思考了半个小时,最后拼了份暴力,$n\le10$ 时暴力做出每一次然后找第 $k$ 小处理,否则直接当做 $k=1$ 去 $O(n)$ 算一遍最小的。 开 D,有点像莫队。思考了一会发现给定 $l$ 是奇数保证了 A 拿的牌都是奇数上的牌,B 拿的都是偶数上的牌,但是博弈本身就不会,所以一点头绪都没有,连暴力都打不出来。 回去看 B,继续思考,直到比赛结束都没看出什么来,最后面向答案直接输出了 $n\le6$ 的答案。 最终得分:$85+5+32+0=122\text{pts}$,啊?A 怎么挂了 $15$ 分? 比赛刚结束时四题难度评的是绿蓝蓝紫,可能是结果太低于预期了,最后四题难度变成了绿紫紫黑,离谱。 由于跟评讲时间冲突,所以没打 AcWing 的周赛,不知道以后还有没有机会打了…… 讲题,A 和我思路一致,看了下别人的实现才发现原来我想复杂了。B 题则证明可以从图论方面去证,但大部分还是考找规律,而找构造题规律就得看和出题人电波能不能对得上了,这次我没有。至于 C 和 D,太难了就不没听,明天补一下题解吧。 晚上打了一场 ABC。 A 题由于没读明白题所以花了点时间,看了样例才搞懂,一分五十才写完,开局已经落后了。B 题由于没想到 10 和 12 这种情况,还花了点时间调试,八分四十五才过掉。C 题一眼莫队板子,啊?后面想了想,发现查询的话前缀和就可以,特判一下端点的情况,树状数组就可以做了。写一半突然发现压根就没有修改操作,直接求前缀和就可以了,浪费了很多时间,十四分才过掉 C。D 题发现很像这周考过的一场模拟赛里面的题目,直接一样的办法用链表维护一下就可以了,二十二分的时候过掉了 D。 E 题发现是模 $k$ 最小生成树,啊?模 $k$ 最短路我知道,但模 $k$ MST 是啥啊,有这东西?后面看到数据范围发现不超过 $28$ 条边,直接状压然后拿并查集判断一下即可,中途用 __builtin_popcount() 剪枝,时间复杂度实际上是组合数级别的,应该没有什么问题,但是我状压不是很熟,稍微写了有一会,四十六分的时候过掉了 E。 开 F,我还是第一次在 ABC 从 A 打到 F,虽然我本来就没打过多少场 ABC。F 题想了很久,总感觉是某种拓扑排序 dp,可能是因为最近做过题面长得比较像的题目。发现加边时会出现需要一整块往上平移的情况,所以想法是用并查集维护相对于集合根节点的大小关系,但是感觉很难维护,所以把路径压缩换成了按秩合并,最后成功维护出来了,跑样例貌似蛮快的,但是交上去之后 TLE 了 6 个点,还 WA 了 8 个点。后面再仔细想想,发现路径压缩是可以做的,只是压缩的时候需要更新一下相对大小关系,改完之后一口气过掉了所有样例,交上去过了,于是我在八十五分半的时候过掉了 F。 剩下 G 题看了一下,只能看出来操作一只有一次以及操作之间顺序无所谓,想法只有枚举第一次操作后的序列然后 $O(n)$ 算总代价,发现不太好算,遂放弃。 这场打完之后 rating 加了 256,算是一点小惊喜吧,白天打比赛打的都挺烂的,晚上总算打好了一会。其实这也是我第一次把 A 到 E 全部做出来,看了一下我的比赛记录,一共九场,rated 的只有八场,之前有一场迟到了所以打的 unrated。我第一场只写了 A,不知道剩下的是不会还是不想打了,performance 甚至不到 200,后面几场基本都是只能做 2-3 题的样子。上周那场 ABC 做出来前四题,E F 都写了但是都挂了,我的 performance 第一次上千。这一场则是直接一路切到了 E,然后 F 一次罚时之后过掉了,performance 到了 1599,排名也是第一次三位数。 感觉进步确实很大,但是可惜不知道以后有没有机会再打了。 睡前随了模板题:裴蜀定理、Nim 游戏、线段树 1。今天没有做题,一整天都在打比赛,也蛮累的。 模板(110/100) 刷题(0/200) 活动(150/100) 今日总分 $260\text{pts}$,溢出 $0\text{pts}$,挂分 $15\text{pts}$,总计增加 $910\text{rp}$。 11 月 12 日 洛谷运势:小吉。今天是周日,好耶。 早上起来先看了下昨天下午的比赛,A 题昨天其实改了但忘记写了。 改了下 B 题,感觉 zig-zag pattern 这个东西很神秘,网上也查不到,不过大致的构造方法我是会了。我发现 B 题我其实可以打 $40$ 分的,但是昨天后面忘了打 $k=1$ 的分了,然后 $n=6$ 输出答案的时候不小心输反了,所以最后只过了第一个点。按照题解的构造方法,很轻易就过掉了。感觉这题其实构造实现很简单,但是构造方法太难了,不过其实貌似也不到紫的难度。 看了下 C 题,昨晚讲完课后发现我写的部分分应该不止 $32$ 分,我猜是我写挂了,重新看了下发现果不其然,有些地方漏开 long long,答案也忘取模了,改完之后就有了 $52$ 分。 这场满打满算我应该能打 $100+40+52+0=192\text{pts}$,已经超过线下联测强校的 $10%$ 分数线了,洛谷上甚至能排到 rk33,如果 B 题电波对上做出来的话就能排在 rk6。不得不说这场比赛虽然难了点,但是质量确实很高! 中午继续补课。由于背包没学好,dp 有点听不下去,打算先补补背包再来听,于是先去听数学部分了。 下午打了场模拟赛。 A 题读完就会贪,但是读完同时也忘了题目保证给的是排列,导致我的做法复杂了,然后不知道什么地方写挂了之后就换了个更复杂的做法。B 题第一眼数学题,想了一会没什么头绪,跳了。看到 C,虽然我看这种多次询问区间的题都感觉像莫队,但是这次我学会了先不看区间问题,考虑对于整个序列做时要怎么做,然后序列上发现就是个 dp,然后联想到了扶苏姐姐讲过的 DDP(虽然这题并不带修),发先 dp 的过程用矩阵可以很容易得表示,所以就把每个位置的矩阵求一下之后用查询区间乘积即可。想到可以前缀和优化,但是不会求逆矩阵,所以只好用线段树来查询区间积。复杂度有 $O(m^2q\log n)$,发现过不去,于是就加了很多优化。最后发现 MLE 了,放弃后面的 $40$ 分然后又优化了一下空间之后才过编,然后非常神奇的一次性过掉了!过掉之后再优化了一点点空间,想着多过一个点也好。然后回去做 B,观察大样例答案发现要么是自己是质数、要么是和相邻的相加为质数、要么把一段前缀加上负数的一半消去后由另外两个数相加的得到质数,推测连续三个数以上的和不可能为素数,那这样的话做法就简单了,筛完质数然后记录一下两种情况,再倒过来推一下每个数到这两种情况的数的间隔,最后应该答案是可以 $O(1)$ 出的,过掉小样例后就交了,发现大样例第一个就错了,但只剩三分钟所以摆烂了。 最后总分:$80+0+0+0=80\text{pts}$。A 题的烂做法挂了 $20$,B 也没过,C 全部 MLE 了,再缩小空间之后有 $30$。这段时间打的最烂的一场。 A 题压根就没那么复杂。B 由于时间剩的不多,然后脑子又特别乱,没有想清楚,所以最后还是写挂了,靠着特判过掉了一个点,但由于 subtask 所以没分。 听了评价,发现我 B 做法是对的,就是没有讨论明白。C 也确实是处理矩阵前缀积和处理逆矩阵前缀积,但我不会。其实这是我第一次写矩阵,我个人觉得已经不错了,就是可惜没拿到分。 晚上随了模板题:裴蜀定理、逆元、字符串哈希、exgcd。全都一次性过掉了,感觉随到的经常重复啊,是不是要考虑 ban 掉一些已经熟到闭着眼就能打出来的模板呢? 今天的做题量很低,比赛也打的很烂,感觉状态不是很好,需要调整,希望明天能好起来。 模板(140/100) 刷题(0/200) 活动(100/100) 今日总分 $240\text{pts}$,溢出 $0\text{pts}$,挂分 $30\text{pts}$,总计增加 $940\text{rp}$。 第三周 11 月 13 日 洛谷运势是大吉,等级分没有更新,咕值貌似也没有更新,无所谓了。 上午有模拟赛。做了两个小时一题都不会,连暴力都写不出来,实在不行了去求助各路 gpt,连清华的 glm 都做不出来第一题,这种题不拿去 CTST,拿来做 NOIp 模拟赛?? 还剩半个多小时的时候放弃了,真的是浪费几个小时的时间。后面看完题解发现自己 A 题想法是正确的,已经转化完了,就差最后的贪心了,但是真的很难想。 感觉自己状态很差,加上现在没人监督很容易颓,必须调整状态,遂战术洗手间。冷静思考了一下接下来 NOIP 考前还有什么要做的: 把洛谷的网课补完,并完成能完成的作业 把 NOIP 前的所有模拟赛题,在我能力范围内的都改一下 整理一下近期完成的题目用到的 trick 补充知识点,尤其是背包问题 把这几年的 NOIP 真题大致看一下 我嘞个豆,这么多? 但是不管怎么样,每天打板子是必须且正确的。随了板子:ST 表、逆元 2、单调栈。然后继续完成图论的倒数第三道作业,写完交上去发现洛谷的 rmj 又寄了,上 at 看是 AC 的。 中午看了下背包 dp,其实很简单,很快就全部理解并记住了。接着又看了树上背包以及区间 dp,还看了几眼 DAG 图上 dp。其实都可以理解,只不过我自己并没有形成观念和套路,打算有空把上面的例题都刷一下。看完之后顺便还看了一下格雷码、约瑟夫问题、悬线法,基本都可以理解。 然后学了下 2-SAT,也不难理解,过掉了模板题。晚上一不小心又刷了会 b 站,注意过来后赶紧回来了。然后听说 at 的 rmj 又暂时好了,赶紧回去交了一发,UKE,再交一发,过掉啦! 晚上继续写题,然后终于完成图论的作业了!感觉最后几题二分图匹配的题真的很妙,但目前还是没那么容易看出来二分图匹配,对于这类二分图或者网络流的建模能力确实不是很好。 看了一下洛谷日报的二维线段树,分为四分树和树套树两种写法,都没写过,最沾边的只写过线段树套平衡树,而且 NOIP 也不考,所以就没仔细学了。 然后写了状压的板子题?其实感觉连我都能做出来的状压基本都很板了(),基本没调就过掉了,感觉最近好稳,码力提高了很多!虽然该挂分的还是会挂。 今天从决定放弃模拟赛开始状态就有回升了,感觉不错!dp 基本上是把提高级范围内的又重新复习了一下,现在感觉就差上手了,打算明天把几道例题过一过。rp 破千了,嘿嘿。 模板(110/100) 刷题(380/200) 活动(50/100) 今日总分 $490\text{pts}$,溢出 $90\text{pts}$,挂分 $0\text{pts}$,总计增加 $1030\text{rp}$。 11 月 14 日 洛谷运势:小吉。 早上看了下教练发来的一套 NOIP 模拟赛题,A 题想了很久,一开始以为是容斥,后来发现是拆贡献,再后来发现不能拆,再后来发现是个完全背包。前前后后两个多小时,结果好不容易写完了,第二个样例都过不去,但第二个样例又很大,手算也算不出来,检查了无数遍代码都没明白错哪了,甚至换了种写法重新写过出来的答案还是跟之前的一样,感觉像是我读错题了。不管怎么说,感觉这个平台的模拟赛一直都很缺德。 由于完全不知道错哪了,感觉状态又变成昨天上午那样了,遂像昨天一样放弃了这场,赶紧调整状态。 然后就开始刷背包的例题,然后很轻松的把 OI Wiki 上给出的各种背包的例题都过掉了。感觉普通的背包 dp 很好想也很好写,基本都可以一次过,应该是掌握的比较好了。 中午上网课,由于现在掌握背包 dp 之后感觉对 dp 理解加深了,遂退回去从第二题开始再看一遍 dp 那一堂课。看完第二题之后就开始自己写,然后一遍过掉了,好稳啊我。 然后就开始打板子,今天随到了拓扑排序、字典树、缩点。这次字典树终于一遍过了,另外缩点写一半突然就晕了,我 SCC 写的好好的,突然把 SCC 当成了 eDCC 以为我自己写错了,然后改成了 eDCC,结果刚准备提交的时候看到记录发现我又记错了。最后又改回来了,还是有惊无险一次性过掉了。 其实讲道理 SCC、eDCC、vDCC 还是很好区分的:SCC 是有向图、eDCC 需要传入反向边,vDCC 需要记录 child。不过有时候就是会突然一下搞混。 闲的没事数了一下,发现绝大部分板子我都敲过一遍了,但是貌似还是有五六个没有敲过,明天开始直接选没敲过的板子吧,就不随机抽取了,毕竟要确保考前全部至少打一遍。 接下来就在一直写作业了,感觉 dp 的作业都很有启发。然后一下午和一晚上,边听课边写题完成了状压 dp 和树上 dp 的部分。而且我发现我写的意外的很稳,几乎都不需要对着主程序调。 感觉今天的状态其实还是很不错的!背包问题练习了,作业也完成了很多,课程进度也追了很多,就是上午浪费了点时间,希望这几天都能像今天下午和晚上这样高效! 模板(100/100) 刷题(570/200) 活动(50/100) 今日总分 $720\text{pts}$,溢出 $320\text{pts}$,挂分 $0\text{pts}$,总计增加 $1350\text{rp}$。 11 月 15 日 洛谷运势:大吉。 上午有场模拟赛。A 题想了一会想出了 $O(n^2)$ 的暴力做法后就去搜原题了,发现今天这场四题都是原题。 倒不是不认真打,但是 NOIP 前也不要求我赛时做出什么题了,现在更主要的是拓宽思路、练习部分分和思考,尤其是不能连暴力都打不出来。 然后 A、B、C 三题都是在想出暴力以后去看题解,并且在看懂的情况下完全由自己写代码,然后过掉了这三题。其实 C 题我已经接近想到正解了,但是我以为区间 dp 没有优化空间了,但事实上是有效状态是 $O(n^2)$ 级别的,所以可以把区间 dp 优化为记搜就可以过掉了。 总分 $100+100+100+0=300\text{pts}$,打得比较高,而且三题我都是最短代码。发现自己其实有优化代码的习惯,而且优化能力很强。 下午继续听课,中间随了板子:欧拉路径、边双。然后看了一下换根 dp 板子题,并且发现了我半年前写的代码中的一处错误,并且以更优秀的实现方式重写了代码。 放学时思考了一下仍未完成的任务: 完成课程及相关作业 看一看这几年的 NOIP 真题 补一补这段时间的模拟赛题 整理 trick 上 OI Wiki 补充知识点 任务很重,时间很紧张。 晚上继续听课,讲完一个知识点就做一下对应部分作业。讲基环树 dp 的时候想着找环比较麻烦,想去题解里学一个比较好的视线,翻来翻去发现没一个写的比我好的。 最后做了一下比较有特殊技巧的题目,比如悬线法、优先队列优化 dp 等等,结束了 dp 部分的课程。 模板(100/100) 刷题(340/200) 活动(100/100) 今日总分 $540\text{pts}$,溢出 $140\text{pts}$,挂分 $0\text{pts}$,总计增加 $1490\text{rp}$。 11 月 16 日 集训今天结束,上午起的很晚,起来后收拾了一下吃了个饭就出发了。 下午有场模拟赛,但是到家的时候模拟赛只剩不到两个小时了。 开题,5 min 会了 A,然后 10 min 切掉了。开 B,给了一堆二元关系,考虑了并查集和图论,最后感觉像是转成图论后用拓扑之类的,但是由于 $n,m$ 很小,很慌。写完之后过了小样例,大样例第一个测试就挂了,但是先交一发再说!开 C,呃,随机增量法?哦好像不对,求的不是同一个东西……不会。开 D,不会。回来写 B,再重新推了一下之后发现之前写的并不完全,改一下之后过了小样例,大样例没测但是不管了直接交。 然后就开摆去吃饭了。最后成绩:$100+16+0+0=116\text{pts}$,全场只有两个做出来 B 的,没有做出来 CD 的,哈哈。 晚上听了讲评,A 题我感觉我的做法比正解更好想。B 题正解是一个很神奇的 trick:当我们只关心数值之间大小关系的时候,我们可以考虑枚举或者二分一个数字,然后把大于等于这个数字的数看做 1,小于这个数字的数看做 0,然后再在 01 串的角度去按照题目 check 或计算。在 B 题中的体现则是:我们需要 check 所有排列($n\le18$),$O(n!)$ 难以接受,但我们可以按上述 trick 之后就可以转变为 $O(2^n)$ 枚举 01 串后对 01 串去 check。问了 mrsrz 老师说这个套路很常见,希望 NOIP 会考。C 题则是扫描线,其中有线段树的高级使用技巧。D 是防 AK 题,能写出来 $O(n^2)$ 的 dp 就不错了。 希望 NOIP 我也能 10min 切掉 A。(考后upd:我去,我 NOIP 真的 10min 切了 A) 晚上刷了板子,随了一道 CRT,然后自己挑了没写过的 KMP 和点双。今天没写题,也没补课,不应该,但是有点忙,实在没办法。 现在任务更紧张了,打算放弃第五节课的数学部分了,相信 NOIP 不会考!明天任务计划就是把去年 T4 写了,然后补一补之前的比赛题,然后整理一下午的 trick。 模板(160/100) 刷题(0/200) 活动(100/100) 今日总分 $260\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $1490\text{rp}$。 11 月 17 日 洛谷运势:中吉。最后一天了。 早上起来之后看了下去年的 T4,想了一会只会 $O(n^2q)$ 的做法,感觉能优化到 $O(nq)$。 毕竟难度还是有的,所以就不考虑死磕了,直接开题解,发现是我没学过的东西,区间历史和是什么?在大佬的博客学了一下吉老师的神仙做法后再回来看就明白多了。 做法大致会了,确实就是查询区间历史和,不过具体式子还没有详细推,放掉这题了。 然后就开始整理 trick 了,把集训期间所有比赛以及洛谷 NOIP 冲刺计划所有比赛和我这三周做的题都看了一下,从上午整理到晚上才算是基本整理完,希望考到的全是上面的。 还看了一些大佬收集的 trick、易错点、骗分等内容。 停课日记正文的最后一篇了,这三周应该是我学 OI 以来进步最大、最认真、水平最高的时间。不祈求超常发挥,不要给自己太大压力,希望明天能不留遗憾就好! NOIP 2023,rp++! 后日谈 感觉一下子,不仅仅是 NOIP,很多事情都结束了。要说的话反而意外地没有那么多,一时间不知道究竟需不需要单独开一篇游记,索性就接着记录于此了。 11 月 18 日 考完 NOIP 回来之后,其实已经陆陆续续在各大估分平台上得知了自己的成绩。所有平台除了 T3 有小浮动以外都一样,大致估分应当如此:$100+100+[10,25]+0=[210,225]\text{pts}$。 考场上还特意用注释标记了时间,所以这里就简单写一段游记吧: 8 点 52 分,开考 20min 时读了一遍题之后切掉了 T1。 T2 从 2-sat 想到并查集想到有向图想到搜索,换了很多做法调了很久,10 点 47 才过掉所有样例,其中最大样例刚测的时候炸了,怀疑爆栈了,然后确实调大了就过了。 此时考试时间刚过去一半,前两题还没拍但是已经过了所有样例。心里想着貌似有机会写出 T3 或者至少剩下两题打个比较优的暴力,于是战术洗手间,让自己冷静一下。 回来继续开 T3,在刚开考读题的时候就已经有过一些思考,所以读完 T3 之后马上就会了判断合法情况的方法,那接下来只要考虑如何进行构造即可。大概是由于这题数据比较难造,我手造的小样例完全没有任何强度,让我误以为可以直接贪心,写了半个多小时之后只过掉了第一个小样例,而第二个样例在手玩两三次之后发现依旧可以直接贪心,所以依旧继续错下去。最后不到一小时的时候赶忙去看 T4。 然而 T4 没有什么头绪,感觉可以 dp,但是 dp 式子貌似很难写。最开始大概糊了个 $O(mk)$ 的做法,发现根本写不出来,一步步退档,最后写了个不知道什么东西,总之只过掉了第一个样例。 回到 T3,这时候决定不要再想正解了,直接注释掉代码开始暴搜,发现连搜都搜不对,很不理解。但此时 T4 又有一个 $O(nk)$ 思路,决定在最后半小时冲 T4 暴力。 到最后依旧没冲出来,依旧只过了第一个样例。最后十分钟回去调 T3 暴力,也没调出来,也只过了第一个样例。收卷最后十秒发现暴力的一处错误,改掉了但没有任何测试。 出考场很平淡也很无奈,AB 过了但没拍,CD 暴力很可能写挂,这就是我最后的结局。 回到家后拿到了代码,测了 C 的小样例对了,松了一口气。另外这三周备赛挺稳的,代码一次性过的概率很高,所以我相信我应该是不会挂分的。那么最终的成绩应该就是看后面两题的暴力究竟能不能打出一点东西吧。 最好的结局就是 CCF 的数据过水,T3 意外拿了比较多的分,然后 T4 也能有一个点的强度跟样例相同导致我的神秘暴力可以骗到一点分。但是这不太可能,而且用处也不大。 有一个学的比我晚几个月的同届也做出了前两题,但是他后面两题挂的很彻底。有两位高三的学长为了 NOIP 冲 1= 也停了一段时间课,但最后的成绩有点遗憾。 其实这次我打的并不差,可能确实到水平上限了。该做出来的题确实做出来,剩下两道也确实不在我的能力范围内,只是两个多小时连暴力都没打好,有些可惜。 这段时间完全荒废了 whk,之后要补的内容实在太多太多。最终考的分数比 CSPS 高,但又没高到哪去。不过还算安慰的是,至少这个赛季始终都是 1= 的。 11 月 19 日 今天上午作为深圳科创之星代表被邀请参加了创客节的活动,看到了很多项目,但其实感觉所谓的 AI 展和我所喜欢和期待的人工智能略有一点区别。仔细想想,其实我想研究的应该叫机器学习或者神经网络,相比于现在所谓的广泛的人工智能,应该要更加理论化,也更贴近与编程。 下午五点,CCF 终于公布了 CSP 获奖名单,而 cz 也马上就公布了今年的钩子线。没想到在超大基数的加持下,今年的七级钩真的只需要 175。 喜提七级钩,蛮高兴的。 11 月 20 日 一觉睡到了将近十二点。 咕值更新了,$322$,排名 $344$。下午获奖记录更新了,也顺利拿到了七级钩。 今晚开始就要回学校回归 whk 了。 11 月 26 日 这一周基本在专心 whk,简单复盘了一下 NOIP,并且至今不会 T3T4,确实是实力问题无法避免,不过暴力写挂了比较可惜。查到了得分:$100+100+10+0=210\text{pts}$,取到了估分区间的下界。 其实打的并不差,在省选中占分比去年高了几分,但还是希望不大,所以决定直接退役了。WC 没必要参加了,APIO 也不打算参加了,NOID 太费钱还一点用没有,说实话这两万给我拿去买好的电脑或者游戏主机岂不是更爽? 打算打一场 THUPC,然后明年争取 NOIP 再考个 1= 就彻底结束了,大学也有点懒得打 AMC 了。 其实以我的实力,THUPC 和网上认识的比较强的选手组队的话有可能可以打进决赛的,但是还是决定跟同学一起组队打,不过说实话进决赛希望不大,但也不重要了。 其实我放得下的,这半年来学算法已经学傻了。打算退役后回归到之前,自己玩玩编程,做做项目,有空打打游戏写写小说,在学校就认真 whk,我一样会过得很好。 虽然这么说,但其实我 whk 不太想跟学校。不知道有没有那种速成班,半年卷完文化课。 停课这段时间天天写日记,其实有点养成习惯了,感觉写的时候回顾一天的历程很放松,最近没有写反而感觉有点奇怪。哈哈,说不定之后我要开始写周记了。 12 月 9 日 去你的 OI,我发现原来我的生活中没有 OI 过的一样很好,我只是想玩电脑罢了。 重新读了一遍日记,发现我之前因为模拟赛里 10min 切了 A,说过希望我 NOIP 也能 10min 切 A,没想到最后成真了,哈哈。 下午回来打了场洛谷月赛,以后可能就这样隔一段时间打打比赛仅仅保持手感就好了。 现在有点尴尬的是,我的博客里专门为题解开了一栏 Solutions,甚至还专门造了个 algorithms tag,然而我才发了四篇题解就 AFO 了,那我这不白搞了? 附录 参考文献 日记应该不会有参考文献吧……? - [ ] 模板(0/100) - [ ] 刷题(0/200) - [ ] 活动(0/100) - 今日总分 $0\text{pts}$,溢出 $0\text{pts}$,挂分 $0\text{pts}$,总计增加 $1490\text{rp}$。 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/10/30
articleCard.readMore

拾光纪行 | CSP-S 2023 复赛游记

前传 随笔杂谈 | CSP-S 2023 初赛游记 游记 day ~-22 初赛回来后,进行了深刻的反思。 我怀疑我的基础并不是很牢,于是开始尝试去打新手村。 直到国庆节前的这段时间其实大部分时间真的去打了,貌似打了近百道,打到 bfs 了,也不清楚具体打到了多少。(upd:后面才知道原来新手村有题单,一直是对着剪贴板做的) day -22~-9 国庆后,新手村就先停下了。 倒不是因为我觉得写简单题太无聊,而是因为我洛谷网校在国庆有课,这段时间基本是在跟着秋令营的课和题单慢慢做,慢慢学。 这让我想起来之前多校联训的课也是要补的,基本上我只跟着学了个莫队后基本一直处在挂机状态了。 (upd:这很不对。我不应该这样,花了钱报了课最后又没听进去,最后没有好结果的还是我自己。) 顺带一提,这段时间还拿了一道黑题的最优解,还写了我的第一道题解 P4480 [BJWC2018] 餐巾计划问题,算是一个里程碑了! 其实这段时间没有专心搞 OI,因为这段时间的最后两天要考月考。我一直认为竞赛和 whk 之前不得不有一个平衡,否则两个都搞不好,而我确实没办法承担丢下 whk 后可能带来的后果(upd:停课真爽,不想上学了)。 不得不分心去复习一下,至少 whk 不能丢下太多,想保持在重点班第 5, 6 名左右的位置已经让我感到吃力了,但最后考试结果还是在接受范围内。 day -8~-1 考完试之后,这段时间基本就在停课了。 最初我还是很认真的一直打模板、补充知识点,到后来又出现老毛病了——越到关头越松懈,反而一天天开始摸鱼越来越频繁了。 最开始只是打打音游(说起来,由于换了个网,打不开之前的 4K 音游了,于是换了个 5K 的音游,之前一直不习惯空格,现在已经能打的比较顺了,fc 了 YOASOBI 的两首曲子的中等难度),后来大家在机房里翻到了不知道什么团体之前使用机房留下来的 cs1.6(这个机房没有冰点还原),于是信竞就真的变成电竞了。 当然我还是有保证一定量的做题的。 貌似我初赛之后也是这个样子的?下次绝对不会再这样了。 这段时间我同时还自己出了四道 ds 模板题,拿来出校测,还是蛮有意思的一次出题体验。当然,为了不让题目太板,我每个都在板的基础上加了点需要思考的内容,同时也给自己拓宽了一点想法(比如线段树新增一种特殊的懒标记、树剖搭配树状数组之类的) 后来因为办的不错,教练要求再加一场,于是从洛谷选了四题我自己做过的感觉比较有学习价值(增加了我的 trick 的那种)的题目。 另外这段期间,做了超多模板题,包括 Nim 游戏、各种基础数据结构和图论,以及补了很多提高级范围内的数论!(但最后如我所料,一点没用上) 到最后一天为止,终于完成了我计划内的所有模板题,包括最长的树剖以及最不熟练的 splay,顺带修了下很久之前写的 splay 的错误然后过了数据加强版。(所以现在我觉得我的 splay 熟练很多了,但做题还是不一定看得出来) day 0 上午又打了一遍 splay,看了一下对拍(直到这个时候我还是觉得我考场上有机会写对拍的)以及各种考前注意事项和骗分导论。 考场还是去年那个,很小,很烂,而且还不能带零食进考场。 开考时解压包密码还输错了一次,属于是开题不利了。 T1 看了几分钟,硬是没看懂题意,不过好在我见过一道类似的题目(给出 $n$ 个约束条件,求出原数的可能种数),还是差不多读明白了(upd:其实并没有)。 后来我注意到,其实只需要 $O(10^5n)$ 枚举然后 check 一下貌似就可以了,换句话说就是时间完全够用,所以立刻就开始动手写了。 然而悲惨的事,我题意有一部分理解错了,我以为他要求 $n$ 个状态都是同时滚两个滚出来的或者都是同时只滚一个滚出来的,这就导致我不仅花费了更多时间,写出来代码连第二个样例都没过,到最后都没想明白为什么第二个样例答案是 $10$。 我一直怀疑那个 $10$ 里面算上了两个状态本身,然而题目说状态本身不正确,所以我一度考虑特判 $n\gt1$ 时算上状态本身,但最后还是没这么写。(upd:谢天谢地我没改) 大约卡了一个半小时后,我决定先停下 T1,先把后面至少写点暴力先,等后面再回来调(upd:根本回不来)。 T2 看到串串题,再见! T3 一看到题面就猜到大模拟了,虽然很烦但技术含量倒不高,有可能可以做,看了眼 T4 估计只写得出暴力,就打算先写一个小时 T3 了。读完题之后发现这个东西很复杂,但是我意外地没有什么阅读上的问题,都能读得懂,心里也大致对结构有一点想法了,于是就火速开始写。 其实我之所以敢冲大模拟而不是像大部分人一样见到大模拟就跳,是因为我一直以为我写过一道紫色大模拟题 [HNOI2004] FTP服务器 - 洛谷,我觉得这道题貌似没有那道难,最难的部分应该就是考察 STL 的应用以及操作三四的字符串处理了,刚好我都有在别的比赛中写过,感觉是能做的。(upd:后面回到家之后才发现其实那道题我还没写,一直躺在题单里面) 刚开始写,我的想法是维护类型的大小和对其要求以及他的成员,用 map 维护名字到类型 id 的对应,而初始时手动添加四个基本类型即可,很快就写完了操作一,而且操作一是一次性过的。 开始写操作二,一样的做法,维护变量的名字、类型和起始地址,用 map 维护名字到变量 id 的对应,很快写完了操作二,而且很神奇的也一次性过了。 由于莫名其妙的居然一点问题都没有,这时候就整个人是非常有信心的状态。 开始写操作三,这个涉及到一个字符串分割,幸运的是我应该是在最近一个月的某场 ABC 或者 CF 上写过这个,简单调试后很顺利的写出来了,剩下的操作其实就是顺着字符串进行一个递推搜索,累加答案,写完了简单调了会之后过掉了第一个样例,并且把第一个样例的每一个成员都做了下操作三,跟我纸上算出来的一模一样。这时候这一题已经写了超过 1h 了。 开始写操作四,这个也是跟上一题一样的思路,顺着地址进行一个递推,然后路上拼接字符串出答案即可。代码写得很乱,但是逻辑上觉得没问题,就这么写下去,最后调完过掉样例时已经过了 1.5h 了。我还对每一个地址都算了一下操作四,也跟我纸上算的一模一样。 这时候我已经过掉了第一个样例,非常的高兴,结果一测第二个样例,fc 出来好几行不同,当时心态都要崩了,写代码写的很乱,很不好调了已经。 幸运的是,仔细看了下第二个样例,发现还是比较简单的,在纸上模拟出来之后对程序进行了调试,发现是在下标计算啊大小对齐之类的问题,以及操作四判空那里漏判了一种情况。调了 20min,过掉了第二个样例。 这时候我高兴的去测大样例,fc 出来满屏的不同,命令行每滚动一行我的心里就像是被捅一刀,但是仔细看后发现我输出了负数!于是返回代码,把所有大小有关的变量开了llu,顺带还发现操作三漏掉了一个跟地址有关的变量没开llu。迅速的修改完之后,三个样例都过掉了!!! 虽然我很高兴,但高兴的同时时间只剩下 30min 了,而我除了 T3 之外只写了一道 T1,而且还没写出来。 这时候决定全力去冲剩下的暴力,T1 就丢掉了,相信能够拿到部分分。开始写 T2,第一个猜想是跟回文有关,似乎处理出每个点的长度为偶数的回文串的长度,然后对连续的回文串做个前缀就可以了,继而想到我一点也想不起来的 Manacher。 考场上几乎认定这是正解了,于是很难过,前几天想学的时候翻 OI Wiki 的时候想补 Manacher 但看到概念又觉得不会考最后还是放掉了,感觉很后悔。 于是我想着没办法了,打个暴力吧。但是可能是因为悲喜交加、时间紧迫、没有零食,导致我考场严重降智,竟然完全没想到枚举左端点然后用栈可以做到 $O(n^2)$ 的暴力,而是蠢蠢的写了一个 $O(n^2)$ 处理出所有偶数回文串后又 $O(n^3)$ 合并的鬼东西,而且还不知道哪里写错了,连第二个大样例都没过。(upd:考后发现是结论假了,合并做法是错误的) 看完 T4,我感觉有点贪心,因此脑子里也闪过二分的想法(因为刚做过一道三分答案后贪心的黑题,现在对于二分三分+贪心这种简单算法可以出难题是深信不疑的),但是很遗憾时间只剩几分钟了,于是我飞速的打了一个骗分,特判了一下 $n=1$ 的情况,其余就输出样例。 最后比赛结束的时候,一度感觉 AFO 了。毕竟本来是为了弥补去年的遗憾,才希望今年复赛拿个 1= 的,结果这次感觉考的比上次还差。 在校车上明白了 T1 我理解错的点了,发现如果我理解正确的话我写代码的时间应该还能短一半,感觉很亏。虽然确实是我的问题,但我还是强烈谴责 CCF 不给大样例,并且第二个样例没有任何解释,我觉得他哪怕给我一个 $10=8+2$ 我都能想出来。顺便我才发现 $n=1$ 的时候答案必定为 $81$,那我相信我能保下这个部分分,不爆零就意味着 NOIp 还有机会。(upd:今年高中生不是有分就行了,好在我最后考的还不算差) 另外 T2 我也觉得我能拿点部分分,不过看到有大佬的线性做法,感觉很震惊,但是看不懂代码! 发现很多人都跳了 T3,并且很多人说自己 T3 挂了,很害怕自己 T3 也挂分。 发现 T4 确实是排序后贪心,我宣布我精神上做出了 T4。 回家路上搞懂了 T2 的线性做法,类似于一个接受回文串的状态机,同时维护了连续回文串的数量(这里还是在坚持我的那个想法),到家后已经十点多了,今天可是周六耶? 考后估分:$30+20+100+0=150\text {pts}$,和去年一样的分数,估计没 1= 了。 后日谈 10.21:回家后拿到代码自测了,T1 莫名奇妙在各大 OJ 都跑到了远超我预期的分数,T2 也很稳定都都反向挂了五分,T3 也运行的很稳没有挂,T4 也稳定的一分都没骗到!目前云斗估分:$90+25+100+0=215\text {pts}$,核桃估分:$80+25+100+0=205\text{pts}$,洛谷和小图灵还没出完整数据。 10.22:洛谷估分:$70+25+100+0=195\text{pts}$,小图灵估分:$80+20+100+0=200\text{pts}$。没想到 T3 意外地很稳,一点都没挂,T4 依旧一分都没骗到。T2 有一点小浮动但和我预期的差不多,而 T1 就是一直在 $[70,90]$ 漂浮不定,基本是决定我这次的分数了。 10.28:这段时间看了很多游记,同时 AK 了洛谷上的自测并写了些题解。看游记的时候学到了很多做法,同时也发现很多人都和我一样 T2 不会栈做法或者推出错误的结论,松了口气。另外看游记的时候又找到俩估分平台,计蒜客估分:$70+15+100+0=185\text{pts}$,信友队估分:$80+25+100+0=205\text{pts}$。然而 CCF 就是不出分,不知道要拖到什么时候。 10.29:昨晚刚说完,今天一早就出分了:$80+25+100+0=205\text{pts}$,跟核桃和信友队上的估分一模一样。至少是上 $200$ 了,松了一口气。虽然我本就发挥的不好,但现在看来还不算很差。 10.30:把这次比赛四道题的题解写完了。听说今年七级钩线很低,那我岂不是还有希望喜提七级钩! 附录 参考文献 CSP-S 2023 Solutions 还有许多其他人的游记 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/10/22
articleCard.readMore

拾光纪行 | CSP-S 2023 初赛游记

关于博客 在这篇游记之前,我想先提一下关于我消失的这大半年。 最后一次发布博客,是在今年一月了。那时候博客刚建立不足一月,又正值寒假,自然更新的很勤快,我相信绝大部分博客都是这样的。 虽然我并没有更新,但我说过,我不会放弃这个博客,这句话到目前为止依旧有效。 实际上,在今年四月份的时候,我还有在编写博客,内容是为Stack模板添加更多界面,实现像本博客一样的题解等功能。但是由于改动部分太多,间隔时间太久导致对GO语言越来越不熟悉,最后便一点也写不出来了。当然,等有空了我会尽量填坑的。 由于我在那之后是专心学习OI,几乎没有碰什么项目,我的模型也停止了训练,最后确实很长一段时间都没有什么能写的内容了。这段时间唯一有在确实的进步的,恐怕只剩下设计了——绞尽脑汁为学校社团设计了张海报,用了许多半调和几何之类的,PS技术也得到了非常大的进步,个人还是非常满意的。(或许我以后有机会可以在博客里发一些设计相关内容?) 另外,很不幸的是,由于服务器ip地址重新分配,导致我的域名解析失效,因此整个网站貌似有几个月的时间只有云盘能上了。当然,目前已经解决了该问题。 在接下来很长一段时间,我都有可能会与之前一样处于没有内容可以更新的状态。不过,我会试着写一些于我而言很有启发意义的题的题解,算是另外一方面的更新吧。 接下来是游记正文。 游记 想了很久,还是决定把游记发在博客而不是题解区里。话说我为什么连CSP-S初赛都想发游记呢?可能是因为我只是单纯的想发一篇游记,顺便提一提我最近的状况。 前言 其实这不是我第一次写游记。去年省选联考的时候我也写了一篇超长的游记,然而D1T1因为返回类型写错导致100pts -> 0pts,省选总分烂到没脸发出来,在游记结尾定的今年的目标看起来也过于笑话,所以并没有发出来。 去年的初赛,我作为一个刚学OI不足三个月的初学者,莫名其妙考了六十来分,还抱回了自己的第一个一等奖。 在这之后,一路学一路比,最后止步于省选,结束了我第一个赛季。 经过剩下半个学期和暑假的努力,我应该是进步了很多的,希望这次能走的更远。 哎呀,怎么前言全在写以前的事呢。 day -7 ~ -2 这段时间一直在认真复习初赛知识点,甚至完全没有摸鱼。 基本上每天一套模拟题或者往年真题,然后其余时间就在b站上看真题解析,或者是给机房其他同学讲题。 在这里感谢一下谷友们总结的初赛知识点提纲,虽然最后究竟考到了多少我也不清楚。 总体来说,刷的真题和模拟题都在七十分以上,自己认为还是很稳的,但是距离目标还是差一点。 基本上是寄希望于考场超常发挥,做的好一点。遗憾的是,这个希望破灭了。 day -1 距离考试只剩下一天,最后做的一套模拟题,虽然主观上很难,但依旧拿了78分。 哟西,很稳。 于是反而更加放松了。由于没有同学需要我解答问题,于是我开始了摸鱼,这时还是简单打打4k的网页音游。 day 0 教练说上午就不刷题了,好好放松一下,于是我名正言顺的坐在机房里摸鱼。打打音游,然后在剪贴板里找到了个洛谷树的游戏,于是一边挂机一边打音游,偶尔给同学讲题,上午大概就这样子过去了。 中午和教练以及同学们一起吃了个午饭,收拾了一下东西,就出发了。 校车好颠簸。 下午以一种神奇但不能说的方式进入了考点。找了一小会考场,发现居然是在电脑室。或许是件好事吧,刷题的时候一直是在电脑上刷的,或许在电脑室的话比较能找到感觉。 他们的电脑室好漂亮,桌子好舒服,就是显示器有点小。 隔壁怎么是个小学生。感觉比我强。感觉被压力了。 开考前,先涂了个什么鬼填涂知情协议书,对我这个填了这玩意儿几年的学生来说自然不是问题。 发试卷,先秒掉第一题,然后看到第二题就感觉不对劲了,越往后面越觉得前几题难度不太对。 糟糕,前15题做太久了。 糟糕,阅读题第一道做太久了。 空调好冷,手好冻。 糟糕,只剩下二十分钟了,我还没写填空。哦豁,完蛋。 后面就疯狂赶进度,十五分钟把填空题做完然后涂卡,至少是在比赛结束前完成了所有内容。 好在最后填空题并不是很难,我也并没有失去很多分。但是,没有以往做模拟题的那种有70分的感觉。 总的来说,还是相信自己不至于初赛都过不了的,但是总感觉没有去年高。出了考场心情算是不太好,感觉发挥的很不好,没有这段时间做模拟题和真题的状态。 不知道是摸鱼过度了,还是不习惯纸质题,但总之我知道肯定到不了我应该考到的分数。 回去的路上,在校车上一直在想:会不会有人比我更惨,准备了一年冲省队,结果初赛翻车连第二把的机会都没有。竞赛这个东西,那么多次比赛,只要失误一次就止步于此了,那样真的会很不甘心。 回家后简单测了一下分数,一开始在计蒜客上测得七十多分,然后在小图灵上是不到七十分,只比去年高了一点点,感觉这次就这样了。虽然发挥失常,但是一等奖还是有的,复赛也不至于进不了。 接下来还是准备好好备赛复赛,在NOIP前展示一下自己的实力,目标是两题半不挂分就行。打算把题库上的提高组的题好好刷一下,以及开始攻略洛谷新手村。 希望自己复赛能发挥的更好吧。 后日谈 09.29 出成绩了,感觉的很准,确实没上70,只有68.5。出成绩前一直担心会不会有涂错卡之类的问题,看到分数跟预估的一模一样才放下心。虽然连一等奖都很稳,但还是觉得没有发挥好。按照平时刷题的状态,这种题应该属于75以上,可以试试冲80的。前面时间花费太多了,可能是我给自己压力太大了,毕竟关系到的太多太重了。不过不管怎么说,稳过就好。 附录 参考文献 无 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/9/16
articleCard.readMore

学习笔记 | MVVM架构中事件(Event)的正确处理方式

前言 最近读到一篇博文,一语惊醒梦中人。虽然早就注意到了这个问题的存在,却对解决方法没有任何的思考,看来我对MVVM的理解依旧不够深。十分感谢博主,学习到了很有用的知识,特记录此笔记。 MVVM中的“事件” 在MVVM架构中,ViewModel通常承担着处理数据的责任,在处理完数据后,将新的UI状态通过LiveData传递给View进行UI更新。其中,LiveData起到了在VM与V中的数据通信作用。 这个逻辑没有任何问题,同时也是目前MVVM的主流和推荐的做法。 然而,相信很多人在使用MVVM架构的过程中都遇到了这个问题:有些时候ViewModel处理完数据后,并不需要更新UI,只不过是想要进行一些需要Context的操作。也就是本文中的“事件(Event)” 例如:弹出Toast、导航、页面跳转。这些或许算UI“操作”,但在我的理解上和UI的“更新”八竿子打不着边。 这时候很容易想到用LiveData来传递……事件? 错错错,大错特错! 事件(Event)和状态(State)的区别 那篇博文关于这里讲的很好,这里简单摘抄一下: 虽然“状态”和“事件”都可以通过响应式的方式通知到 UI 侧,但是它们的消费场景不同: 状态(State):是需要 UI 长久呈现的内容,在新的状态到来之前呈现的内容保持不变。比如显示一个Loading框或是显示一组请求的数据集。 事件(Event):是需要 UI 即时执行的动作,是一个短期行为。比如显示一个 Toast 、 SnackBar,或者完成一次页面导航等。 我们从覆盖性、时效性、幂等性等三个维度列举状态和事件的具体区别 状态 事件 覆盖性 新状态会覆盖旧状态,如果短时间内发生多次状态更新,可以抛弃中间态只保留最新状态即可。这也是为什么 LiveData 连续 postValue 时会出现数据丢失。 新事件不应该覆盖旧事件,订阅者按照发送顺序接收到所有事件,中间的事件不能遗漏。 时效性 最新状态是需要长久保持的,可以被时刻访问到,因此状态一般是“粘性的”,在新的订阅出现时为其发送最新状态。 事件只能被消费一次,消费后应该丢弃。因此事件一般不是“粘性”的,避免多次消费。 幂等性 状态是幂等的,唯一状态决定唯一UI,同样的状态无需响应多次。因此 StateFlow 在 setValue 时会对新旧数据进行比较,避免重复发送。 订阅者需要对发送的每个事件进行消费,即使是同一类事件发送多次。 看到这里,你是不是也和我一样突然恍然大悟? 传递事件的正确姿势 EventBus、Flow都可以用于处理事件,然而历史悠久的EventBus虽然业界推崇但太难还有点小题大做、未来主流的Flow虽然谷歌推荐但太新不会用还有点看不懂。 就想用LiveData,怎么办? 这里贴一个相对完善的解决方案: open class LiveEvent<T> : MediatorLiveData<T>() { private val observers = ArraySet<ObserverWrapper<in T>>() @MainThread override fun observe(owner: LifecycleOwner, observer: Observer<in T>) { observers.find { it.observer === observer }?.let { _ -> // existing return } val wrapper = ObserverWrapper(observer) observers.add(wrapper) super.observe(owner, wrapper) } @MainThread override fun observeForever(observer: Observer<in T>) { observers.find { it.observer === observer }?.let { _ -> // existing return } val wrapper = ObserverWrapper(observer) observers.add(wrapper) super.observeForever(wrapper) } @MainThread override fun removeObserver(observer: Observer<in T>) { if (observer is ObserverWrapper && observers.remove(observer)) { super.removeObserver(observer) return } val iterator = observers.iterator() while (iterator.hasNext()) { val wrapper = iterator.next() if (wrapper.observer == observer) { iterator.remove() super.removeObserver(wrapper) break } } } @MainThread override fun setValue(t: T?) { observers.forEach { it.newValue() } super.setValue(t) } private class ObserverWrapper<T>(val observer: Observer<T>) : Observer<T> { private var pending = false override fun onChanged(t: T?) { if (pending) { pending = false observer.onChanged(t) } } fun newValue() { pending = true } } } 该解决方案最早由 Jose Alcérreca 给出了思路和基本框架,经过了许多大佬修改和完善,使用方法也大致和LiveData相同。 有一点比较重要的是,LiveData数据是粘性的,无论何时订阅都会立即触发一次 onChanged()。对于状态来说,订阅时立即获得最新的状态,这没有问题;然而对于事件来说,并不需要获得订阅前的上一个事件。 变量 pending 在这里正是用于解决这一问题的。在初始化时 pending = false,使得在刚订阅时不会立即触发 onChanged()。 相似解决方式有很多,例如美团和这篇文章的方案都是设置一个版本标识符,初始化时等于LiveData的version,再通过判断新接收的数据的version是否大于存储的version来分辨是不是上一次的数据。 另外,该文章中也提供了使用LiveData实现事件总线的思路和例子,可以参考。 不过我并不希望在 ObserverWrapper 中持有LiveData对象,因此推荐使用 pending 做法。 因此,实际上,这个方案与LiveData的区别只是非粘性而已,对 postValue 丢失数据的问题并没有进行处理。方法当然也是有的,我参考这篇文章对该方案进行了修改,增加了一部分内容以避免丢数据: open class LiveEvent<T> : MediatorLiveData<T>() { ... private var mainHandler: Handler? = null override fun postValue(value: T) { if (mainHandler == null) { mainHandler = Handler(Looper.getMainLooper()) } mainHandler!!.post(Runnable { setValue(value) }) } ... } 那么,这一段代码为什么会生效呢?这个就要涉及到LiveData的源码了,可以看这篇文章,里面对这两个方法的源码进行的分析和解释。 简单来说,postValue 其实就是使用了一个线程锁(这也是 postValue 高并发时丢失数据的原因),使用Handler切换到主进程后调用setValue 来设置数据。 因此,我们只需要重写 postValue,收到新数据一律直接抛到主线程 setValue 即可。 现在,我们已经有了一个简单好用、不会丢数据、非粘性、可感知生命周期、可观察的事件对象了。 最后,贴上总代码: LiveEvent.kt: import android.os.Handler import android.os.Looper import androidx.annotation.MainThread import androidx.collection.ArraySet import androidx.lifecycle.LifecycleOwner import androidx.lifecycle.MediatorLiveData import androidx.lifecycle.Observer import kotlinx.coroutines.Runnable open class LiveEvent<T> : MediatorLiveData<T>() { private val observers = ArraySet<ObserverWrapper<in T>>() private var mainHandler: Handler? = null @MainThread override fun observe(owner: LifecycleOwner, observer: Observer<in T>) { observers.find { it.observer === observer }?.let { _ -> // existing return } val wrapper = ObserverWrapper(observer) observers.add(wrapper) super.observe(owner, wrapper) } @MainThread override fun observeForever(observer: Observer<in T>) { observers.find { it.observer === observer }?.let { _ -> // existing return } val wrapper = ObserverWrapper(observer) observers.add(wrapper) super.observeForever(wrapper) } @MainThread override fun removeObserver(observer: Observer<in T>) { if (observer is ObserverWrapper && observers.remove(observer)) { super.removeObserver(observer) return } val iterator = observers.iterator() while (iterator.hasNext()) { val wrapper = iterator.next() if (wrapper.observer == observer) { iterator.remove() super.removeObserver(wrapper) break } } } @MainThread override fun setValue(value: T) { observers.forEach { it.newValue() } super.setValue(value) } override fun postValue(value: T) { if (mainHandler == null) { mainHandler = Handler(Looper.getMainLooper()) } mainHandler!!.post(Runnable { setValue(value) }) } private class ObserverWrapper<T>(val observer: Observer<T>) : Observer<T> { private var pending = false override fun onChanged(t: T?) { if (pending) { pending = false observer.onChanged(t) } } fun newValue() { pending = true } } } 当然,建议还是至少要了解一下被谷歌十分看重并当做未来主流来提供全方位支持的Flow,相比LiveData更贴合Kotlin的协程,非常强大。 关于读写权限 在MVVC架构中,LiveData应该只能由ViewModel持有和写入,而View层则应该只读,所以我们经常会有以下写法: private val _name = MutableLiveData<String>() val name: LiveData<String> = _name fun somefunction(){ ... _name.postValue("失迹") ... } 这个想法和写法并没有问题。将具有读写权限的MutableLiveData设为私有成员,只对View暴露只读的LiveData。 此写法更为规范,但仅仅是为了防止你在View和Observe中误修改,就需要增加代码的数量和降低阅读性,我个人是极不情愿的。 就如同约束性的问题一样,我们真的有必要为了防止可以避免的失误而增加多余且无实际作用的内容吗? 或许我们需要照顾不懂操作的用户,但我们有必要考虑其他不懂规范的程序员到这种程度吗? 话题扯远了。因为LiveEvent本质上就是个没有粘性、不会丢数据的LiveData,因此自然是支持以上写法的: private val _showToast = LiveEvent<String>() val showToast: LiveData<String> = _showToast fun somefunction(){ ... _showToast.postValue("如你所见,这是一条吐司") ... } 这样写的话最后在类型上又会转回LiveData,不过没有影响,当做事件用即可。 对于跳转页面、弹出Snacebar之类的事件我们无可奈何,但除此之外,我们应当避免其他的事件。将数据处理为UI状态,才是MVVM的正确实践! 附录 参考文献 Jetpack MVVM 七宗罪之四: 使用 LiveData/StateFlow 发送 Events Jose Alcérreca , LiveData with SnackBar, Navigation and other events LiveData setValue和postValue的区别及详解 把 LiveData 用于事件传递那些坑 基于LiveData实现事件总线思路和方案 使用 Architecture Component 实现 MVVM 的正确姿势 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/1/27
articleCard.readMore

踩坑记录 | Android中LayoutInflater读取attr颜色属性报错:Failed to resolve attribute at index 77: TypedValue{t=0x2/d=0x7f030107 a=-1}

踩坑 背景 项目要适配夜间模式,因此要添加深色主题(Dark Theme),也就是Android开发中很常见的开发要求——多主题。 在查找资料的时候发现多主题似乎在select和drawable上存在很大的坑,不过这里估计是因为使用了Androidx、AppCompat、Material3,所以并没有跳进去他们的坑,反倒是自己找了一处偏僻的小坑钻进去了。 发生这个问题的时候,我正在尝试使用Databinding获取自定义的Toast布局: val binding = ToastBinding.inflate(LayoutInflater.from(context)) 在 attrs.xml 添加自定义属性、在 theme 中赋值、在布局和 Drawable 中引用。 values/attrs.xml: <resources> <attr name="toastColor" format="reference|color" /> ... </resources> values/themes.xml: <resources> <style name="AppTheme" parent="Theme.Material3.Light.NoActionBar"> <item name="toastColor">@color/toast_light</item> ... </style> <style name="Theme.MyApp" parent="AppTheme" /> </resources> drawable/check_circle.xml: <vector android:autoMirrored="false" android:height="24dp" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> <path android:fillColor="?attr/toastIconColor" android:pathData="M10,17l-5,-5 1.41,-1.41L10,14.17l7.59,-7.59L19,8l-9,9z"/> <!-- 这是一个svg格式的矢量图,重要的是↑这个fillColor里引用了attr,后面的pathData是图片绘制路径 --> </vector> layout/my_toast.xml: <?xml version="1.0" encoding="utf-8"?> <layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools"> <androidx.constraintlayout.widget.ConstraintLayout ... android:background="?attr/toastColor" > <ImageView ... android:src="@drawable/check_circle" /> <TextView ... android:textColor="?attr/toastTextColor" /> </androidx.constraintlayout.widget.ConstraintLayout> </layout> 编译时一切顺利,然后一运行,问题就嘎嘎找上门了…… 问题描述 正如题目所说,运行时报了以下错误: Failed to resolve attribute at index 77: TypedValue{t=0x2/d=0x7f030107 a=-1} 一开始我以为这里的 attribute 是属性的意思而不是特指 ?attr/ ,看了半天没看懂,于是上网一顿猛搜,发现跟Theme的自定义属性有关。 然而查来查去,都没有解决办法,勉强知道问题是出在控件没有正确地获取到 ?attr/ 的值。 然而我适配多主题,靠的就是 attr 呀,那不完蛋了? 解决方法 在查找资料的过程中,我发现其他发生相似问题的,大多是 Application 没有设置好Theme、Activity 没有设置好Theme、Fragment 没有设置好Theme。一句话总结相同点,就是用于 LayoutInflater.from(context) 的 context 没有正确地配置Theme。 在尝试过各种办法都无效过后,走投无路的我大胆的猜测:难道是因为这个 View 没有配置好Theme吗? 于是我在 layout/my_toast.xml 的根布局的属性中加上了一行: android:theme="@style/Theme.MyApp" 也就是变成下面的这个样子: ... <androidx.constraintlayout.widget.ConstraintLayout ... android:theme="@style/Theme.MyApp" android:background="?attr/toastColor" > ... </androidx.constraintlayout.widget.ConstraintLayout> </layout> 然后问题就神奇的解决了。 既没有在 inflate 的时候报错,Drawable 也正常显示出来了。 分析原因 似乎由于传入 LayoutInflater.from(context) 的 context 是一个我初始化时传入的 Application 对象,因此布局的View并没有自动获取到 AndroidManifest.xml 中的Theme配置。 而我们知道,attr 是与当前的 theme 完全关联的。 你当前是什么Theme,就获取到对应的 attr 。而如果当前的Theme使用的是默认的Theme或其他Theme,而这个Theme中并没有为对应的 attr 赋值,自然而然就要报错或者出现显示问题了。 总结 attr 是与当前的 theme 完全关联的。出现此问题,大概率是因为用于 LayoutInflater.from(context) 的 context 没有正确地配置Theme。 在对应的xml文件的根布局加入: android:theme="@style/配置了对应attr的Theme" 应该能解决大部分同类情况。 附录 参考文献 说实话,看了很多博客之后才使我受到启发找到解决方法,但是还是感谢他们。虽然他们现在没有帮助我解决问题,但也使我在未来不会遇到同样问题或是遇到了能够解决。 看过的实在太多,但都没有实质性帮助,这里就不列参考文献了。 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/1/22
articleCard.readMore

随笔杂谈 | 2023 新春快乐!

今天是这个小博客成立的第七天,也是这个小博客迎来的第一个春节! 祝大家新的一年里万事如意,目标全达成,2023新春快乐!

2023/1/21
articleCard.readMore

踩坑记录 | Gradle7.0+仓库配置踩坑:Build was configured to prefer settings repositories over project repositories but repository 'maven' was added by build file 'build.gradle'

踩坑 问题描述 新版Gradle从7.0版本开始,新增了 dependencyResolutionManagement 作为推荐的配置仓库方法。 Android Studio也不知道从哪个版本开始,创建新Project时开始使用这种方法作为默认配置仓库的方法,然而这个代码块是放在 settings.gradle 里面的。 这就导致了我想要配置镜像时翻遍了两个 build.gradle 找不到配置仓库的地方,自己加又一直报错: Build was configured to prefer settings repositories over project repositories but repository 'maven' was added by build file 'build.gradle' 意思是已经加入过这个仓库了,然而提示的重复位置却不是默认的配置而是我们自己加入配置的位置,完全不知道重复在哪了,令人摸不着头脑。 解决方法 删除掉其他地方的 repositories,在 settings.gradle 下配置仓库。 顺便附上我自己修改后的镜像代码: settings.gradle: pluginManagement { repositories { gradlePluginPortal() maven { url "https://maven.aliyun.com/repository/google" } maven { url "https://maven.aliyun.com/repository/public" } } } dependencyResolutionManagement { repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) repositories { maven { url "https://maven.aliyun.com/repository/google" } maven { url "https://maven.aliyun.com/repository/public" } maven { url 'https://www.jitpack.io' } } } rootProject.name = "Sample" include ':app' 附录 参考文献 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/1/17
articleCard.readMore

建站技术 | 为自己的网站使用Google Analytics统计

须知 你应当知晓:在中国大陆境内的服务器使用Google Analytics是不合法行为。 《中华人民共和国个人信息保护法》 第三十六条:国家机关处理的个人信息应当在中华人民共和国境内存储;确需向境外提供的,应当进行安全评估。安全评估可以要求有关部门提供支持与协助。 第三十九条:个人信息处理者向中华人民共和国境外提供个人信息的,应当向个人告知境外接收方的名称或者姓名、联系方式、处理目的、处理方式、个人信息的种类以及个人向境外接收方行使本法规定权利的方式和程序等事项,并取得个人的单独同意。 本教程仅适用于满足使用条件的网站,如果你在不满足使用条件的情况下参考本教程使用Google Analytics,本教程不承担任何责任。 是什么?为什么?怎么做? 关于Google Analytics的详细介绍可以参考这位博主的文章:谷歌流量分析工具Google Analytics使用方法指南教程 本教程只记录步骤而省略一部分解释。 如何使用Google Analytics 注册一个Google账号(需要科学上网) 打开 Google Analytics官网(需要科学上网),点击开始衡量 开设账号 填自己的账号名 填写账号名 名称取自己好认的即可,时区和人民币按默认的。 取名 如实填写即可 行业与企业的填写 完成后点击创建,会弹出一个弹窗问你是否同意协议,勾选同意后选择我接受,完成账号创建 同意协议 完成后会自动跳转到刚刚新建的媒体源,选择平台为网站 选择平台 设置网站和数据流的名称,然后点击创建数据流,注意http和https不要选错,网站网址也不要写错 设置名称 接下来设置统计代码 在数据流详情页面点击查看代码说明 查看代码说明 如果是使用CMS来构建网站的话使用第一项可以快捷设置,这里我是静态网页,因此点击手动添加,然后点击代码块右上角复制代码 复制代码 将代码复制到你的网站源代码的 body 标签中 添加代码 重新部署至服务器,统计就会生效了 查看你的网站数据,具体面板详细介绍可以看谷歌流量分析工具Google Analytics使用方法指南教程,上面讲的很详细 至此,你已成功为你的网站使用了Google Analytics。 注意 根据官网:数据可能最多需要 24 小时才会显示在您的 Google Analytics(分析)帐号中。 我这边将近十分钟就有数据,如果你等太久也没有看到统计,建议F12检查一下统计代码是否生效。 附录 参考文献 中华人民共和国个人信息保护法 谷歌流量分析工具Google Analytics使用方法指南教程 版权信息 本文原载于reincarnatey.net,遵循CC BY-NC-SA 4.0协议,复制请保留原文出处。

2023/1/16
articleCard.readMore

建站技术 | 使用 Hugo + Stack 简单搭建一个博客

前言 相信很多人都跟我一样,搭建博客后的第一篇博文便是教如何搭建博客。就像是在第一次坐飞机打电话教你如何购买飞机票,甚至路上还一边走一边记录了去往飞机场的地图。 明明自己也是刚刚才搭建好甚至第一次搭建,或许对于博客而言,第一篇博文记录这个博客的诞生过程是相当有意义的事吧。 2024/07/07 更新:本博客受到许多人的参考,当初帮助其他人的目的已经实现,由衷感到高兴。在本文完成后,又对本博客所使用的 stack 进行过一次大改,但未能及时记录下来,时隔久远已无法记得当初修改的内容以及对 golang 语言也变的陌生,所以很遗憾本文应该不再更新。已修复文章中的排版错误(例如空格与标点的使用等),若博客中存在错误,欢迎联系指正。 使用 Hugo + Stack 搭建博客 什么是 Hugo? Hugo 是由 Go 语言实现的静态网站生成器。简单、易用、高效、易扩展、快速部署。 在本地安装 Hugo 拓展版 可以使用官方文档中的安装方式,需要按照拓展版,即名字后带 -extended 的版本。 这里我是 Win10 系统,所以使用官方推荐的 Chocolatey 安装方式: choco install hugo-extended Chocolatey 是一个 windows 系统的软件包管理器,类似于 linux 的 apt-get(不过 choco 不是自带的,也远没有 apt-get 那么常用),算是 windows 码农的老朋友了。如果没有安装 choco 的可以使用其他安装方式或者自行搜索学习使用 choco。 安装完成后,你可以输入以下命令检查 hugo 的版本: hugo version Hugo 拓展版的安装 如果版本后面有 +extended 就说明安装对了(我的版本可能不一样,用最新的版本即可)。 用 Hugo 创建博客 hugo 是一个静态网页生成器,简单来讲,hugo 负责的是把内容填到模板里,而具体生成什么网站则要看你是什么模板。 因此,要用它来生成博客需要用到对应的博客模板,你可以在皮肤列表或者 github 上查找模板,这里除了博客以外还有许多个人主页、导航站、文档页面等的模板。 这里我原本想使用的是 Stack 模板,比较符合我的审美需求,然而功能略有不足,因此我这里使用的是大佬美化后的版本 Mantyke/Hugo-stack-theme-mod。 2025.7.21 Update: 该库目前已被删除,我已将备份上传至网盘,点击这里可下载。我更加推荐直接使用原版 Stack 模板(目前已更新地更为完善,功能更丰富)。 使用这个版本必须安装 hugo 的拓展版,否则会报错: Error: Error building site: TOCSS: failed to transform "scss/style.scss" (text/x-scss): this feature is not available in your current Hugo version 这个 mod 版增加了一些实用的功能和设计,虽然我个人更喜欢原版的左侧栏样式,但除此之外这个 mod 版都比原版要好。 由于 mod 版直接提供了修改好的版本,这里可以直接 clone 或者下载到本地解压即可作为网站启动。正常创建方法可以参考hugo官方中文指南和其他的教程。 按照 README 的使用说明修改配置和内容。 注意 footer.html 中的那个 2006 不要修改!这个 2006 是 golang 的基准时间,如果改了的话时间显示就会出问题。在 config.yaml 中配置好建站年份即可,当前年份是会自动获取的。 修改好后,在项目根目录运行以下命令: hugo server 如果没有报错的话,就可以看到 hugo 服务器在本地运行了。 本地运行 可以打开返回的网址(默认是 localhost:1313),就可以看到自己的 demo 站了。 hugo 生成速度极快,修改完成后几乎可以同时更新,调试和测试很方便(这也是大多数人选择 Hugo 而不是 Hexo 的原因)。 意外发现全屏模式下超级好看! 你可以在根目录创建一个批处理脚本,用来快速启动本地服务器并使用 chrome 打开网页。 本地运行.bat @echo off echo 【本地运行】 start chrome http://localhost:1313/ hugo server pause 修改模板 我希望点击头像的时候跳转到我的个人主页。 \layouts\partials\sidebar\left.html: <!-- 第 12 行修改为自己的网址 --> <a href="https://reincarnatey.net/"> 居然没有配置 favicon? \config.yaml: # 第 30 行修改为: favicon: \favicon.ico 随后将你的 favicon.ico 放在 \static\ 下,重启 Hugo 服务器即可看到图标已经应用了(貌似不重启就看不到)。 友链页面还是略有问题的: 左侧导航栏图标错误 使用小写开头,格式不统一 友链页面不显示回复 \content\page\friends\index.md: # comments: false # 这个 comments 无论是 false 还是 true 都会导致没有回复功能,注释或删除这个属性即可 # name: friends # icon: home # 修改为 name: Friends icon: link # 除此之外,你的友链页面内容也在此文件编写,建议提供友链格式和己方友链信息等。 除此之外,左侧导航栏的间距过大导致出现滑动条,不喜欢的可以调小一点: 间距 \assets\scss\partials\menu.scss : 第 157 行:间距①,即导航栏与上方社交媒体之间的间距 第 166 行:小屏模式下的间距②,即导航项之间的间距 第 169 行:大屏模式(即正常模式)下的间距②,建议设成与 166 行一样的间距 我使用的分别是 0px 5px 5px,可以根据自己的审美和导航项数量来设置。 添加自己的社交媒体/联系方式/RSS stack-mod 默认只有 github 和 twitter,显然不太够。而且尽管没人会用 RSS,但是既然stack 有这个功能那又有什么理由不加上呢。因此,这里需要来添加自己的链接。 首先需要下载自己需要的图标,这里推荐阿里的 iconfont。 搜索图标时建议选择线性,尽量选择粗细一致的,保持图标风格相同,比较美观。 选择自己需要的即可,不用太多,我这里选择了 bilibili 和邮箱,下载时选择 svg 格式,颜色设置为 #2c3e50,大小设置为 24,然后下载即可。(RSS 的图标自带有) 选择需要的图标 将下载下来的图标放到 \assets\icons\ 文件夹下,然后修改社交媒体的配置。 \config.yaml: social: - identifier: github name: GitHub url: https://github.com/Mitr-yuzr weight: 1 params: icon: brand-github - identifier: bilibili name: Bilibili url: https://space.bilibili.com/310707183 weight: 2 params: icon: bilibili-line - identifier: twitter name: Twitter url: https://twitter.com/kkkkkorita weight: 3 params: icon: brand-twitter - identifier: email name: Email url: mailto:korita@reincarnatey.net weight: 4 params: icon: email - identifier: rss name: RSS url: /index.xml weight: 5 params: icon: rss weight:权重,决定排序的顺序,越小越靠前。 icon:图标,即你放入 \assets\icons\ 的文件的文件名(不带扩展名) 正常显示的亮色模式图标 不过这样还是会有问题: 错误显示的暗色模式图标 可以看到在暗色模式下并没有正常显示。这是因为我们下载时指定了颜色,而图标内部指定的颜色是要比样式表优先的,因此导致夜间模式时切换颜色的样式没有生效。 我们右键编辑这两个图标,然后将所有的 #2c3e50 都替换成 currentColor,也就是使用样式设置的颜色。 现在图标在暗色模式下就可以正常显示了。 正常显示的暗色模式图标 添加代码高亮 默认是没有开启代码高亮的,这种东西自然是要开启的。 config.yaml: highlight: style: onedark 添加一个 style 属性,其他全部删掉,就可以显示代码高亮了。你可以在 Chroma Playground 找你自己喜欢的高亮主题,但真没几个好看的,挑了个 onedark 勉强凑合着用吧。 博客评论系统的使用 博客中的评论系统是很重要的。这里我使用的是 Stack 模板支持的 Waline,搭建很简单,可以参考 Waline官方指南。 配置 Waline 则可以参考这篇博客 hugo:添加评论功能(Waline) - 建站指南,我的Waline配置: \config.yaml: comments: enabled: true provider: waline waline: # 这里填你的vercel服务器地址。 # vercel自定义域名会和cloudflare会冲突导致无限301,所以干脆直接用送的域名了 # 注意要部署总域名,而不是最新部署的版本域名(中间有一段随机英文字符的),否则会报 401 Unauthorized serverURL: https://waline-你的用户名.vercel.app/ lang: zh-CN # 文章浏览量统计,在新版waline中已更名为pageview属性,貌似用不了了 # 填入false代表不启用,填入字符串的时候会作为css选择器 visitor: false # 头像来源,在V2中已移除该属性 avatar: emoji: - https://npm.elemecdn.com/@waline/emojis@1.1.0/bilibili - https://npm.elemecdn.com/@waline/emojis@1.1.0/bmoji - https://npm.elemecdn.com/@waline/emojis@1.1.0/weibo # 回复时必填的内容 requiredMeta: - name - email - url # 评论框的默认的文字 placeholder: 欢迎留下宝贵的评论! # 自定义语言设置,参考https://waline.js.org/cookbook/customize/locale.html#locale-%E9%80%89%E9%A1%B9 locale: admin: 站长 sofa: 还没有人评论哦!快来抢沙发吧~ 评论系统 另外 emoji 官方提供了许多预设,可以查看预设列表自行选择。 但是官方提供的预设使用的 unpkg 经常会被墙而导致无法正常访问,这里推荐替换为饿了么提供的国内镜像(将 unpkg.com 替换为 npm.elemecdn.com) 自定义表情 如何编写博客 正如同其他的博客,使用 Markdown 语言来编写博客。Markdown流行且极易上手,因此这里就不多介绍语法,如果不会的可以自己搜索了解。 使用 Hugo 创建文章 在博客根目录下运行: hugo new post/untitled.md 为什么要用 hugo 来新建而不是创建一个 .md 文件呢?这是因为使用 hugo 创建会自动使用已填入 Front Matter的模板。 Front Matter 用于标识文章的标题、时间等信息,hugo 就是据此来生成静态页面。关于属性的含义和用法可以参考 Hugo中文文档。 模板可以在 \archetypes\default.md 下找到: --- title: "{{ replace .Name "-" " " | title }}" # 标题,创建时自动填充 description: # 文章简介 date: {{ .Date }} # 日期,创建时自动填充,格式同 2023-01-15T12:00:00+08:00 image: # 文章的封面,留空就是没有,填文章所在位置的相对地址,通常放在同目录下, math: # 是否启用 KaTex,填 true 启用 license: # 文章尾部显示的协议,false 为隐藏,其他作为内容,留空就是使用 config.yaml 里默认的 hidden: false # 是否隐藏,一般用不到 comments: true # 因为 bug 所以这个属性只要存在,不管是 true 还是 false 都会导致回复无法显示,需要删掉 draft: true # 是否为草稿,建议改为 false 或者删掉这个属性以防止忘记修改,毕竟我们一般都是写好了才部署到服务器上 --- 为了方便,我参考网络以及stack-mod的功能对模板进行了一些改造: --- title: "{{ replace .Name "-" " " | title }}" slug: "{{ replace .Name "-" " " | title }}" description: date: "{{ .Date }}" lastmod: "{{ .Date }}" image: cover.png math: license: hidden: false draft: false categories: [""] tags: [""] --- ## 附录 ### 参考文献 ### 版权信息 本文原载于 [reincarnatey.net](https://blog.reincarnatey.net),遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。 因此我们可以编写一个批处理程序来快速帮我们生成文章: 创建文章.bat: @echo off echo 【创建文章】 set /p input= 请输入Slug: hugo new post/%Date:~0,4%/%Date:~5,2%%Date:~8,2%-%input%/index.md pause 用此批处理程序生成的文章会创建在 \content\post\2023\0115-test\index.md,便于我们整理文章资料,同时后续在同目录下存放文章的封面也不会导致内容很乱。 使用 Hugo 创建类别、标签 创建 Categories 和 Tag 也同理: hugo new categories/testcat.md hugo new tags/testtag.md hugo 会自动应用 \archetypes\categories.md 和 \archetypes\tags.md 的模板,但是这两个模板都不太好,建议都改为: --- title: "{{ replace .Name "-" " " | title }}" slug: "{{ replace .Name "-" " " | title }}" description: image: cover.png style: background: "#2a9d8f" color: "#fff" --- 创建类别.bat: @echo off echo 【创建类别】 set /p input= 请输入类别名: hugo new categories/%input%/_index.md pause 创建标签.bat: @echo off echo 【创建标签】 set /p input= 请输入标签名: hugo new tags/%input%/_index.md pause 注意:如果创建多级文件夹时文章文件名不是 index.md 或者类别、标签文件名不是 _index.md 的话,设置封面图片会出现问题。 Markdown 编写工具的安装 生成之后,我们就要编辑文章了。 实际上我也是第一次在本地写博客,此前并没有安装专门的 Markdown 编辑器,基本是手边是啥就用啥(比如 VS、IDEA、我不记得名字的某个云笔记、Notepad++、甚至 windows自带的记事本)。 毕竟之前只是为项目编写 README,够用就行了,但对于博客来说这样肯定是不满足我们日常编写的需求的,因此推荐安装一个 Markdown 编辑器。 曾经听过 Typora 的大名,然而不幸的是该编辑器已经转为收费且优化不足,因此这里推荐免费开源且好用的 MarkText。 安装方法很简单,release 里面下载对应的版本安装即可。不过在我写这篇博客的时候 MarkText 还没有做好国际化的准备,因此尽管已经(看起来)有中文语言文件却无法切换语言。 这里推荐一个大佬的汉化版本 marktext-chinese-language-pack,可以直接在该仓库的 release 仓库里下载对应版本的压缩包,替换到原版安装目录就好了(或者不安装原版直接使用汉化的程序文件,应该是直接可以用的,不过我比较喜欢通过官方安装器安装一遍以免注册表或者启动栏之类的问题)。 打开之后应该就是中文了,简单设置后直接输入便可以开始进行内容的编写了。 MarkText 中图床相关设置 在文章中图片是少不了的,那么图片该放哪呢?虽然你可以放在自己服务器上,但这样会占用服务器资源、增加请求数,还增加了你的部署成本和时间。 因此,在编写博客的时候,我们通常会需要一个图床来存放那些不是特别重要的图片,例如过程的截图等。 曾经最常用的微博图床现在已经开了防盗链,其他图床大多要钱或是国内无法访问,难以找到稳定的图床。因此有能力建议自己搭一个,并不是很难。 这里我使用的图床上传工具是 PicGo,虽然 MarkText 支持调用 PicGo 进行上传,不过由于对接的不是很好,目前使用上还是会有问题,不过直接在 PicGo 软件内进行上传已经足够方便了。 使用方法很简单,安装完成后按照这里获取 Token 在图床设置->SM.MS 中进行配置即可。之后就可以在上传区上传图片文件或是截图后点击剪贴板图片快速上传了。默认会在上传完成后自动复制 Markdown 格式的链接,直接复制到编辑器就好了。 由于 sm.ms 最近经常被墙,建议在备用上传域名一栏填上 smms.app,这是官方的镜像(?)地址。 镜像设置 MarkText 中编辑图片的标题 在我们的博客中,有时会希望在图片下方显示一个标题。在 MarkText 中因为会直接渲染文件,这时需要在修改页面切换至完整模式。(不是很方便,这算是缺点吧) 选择编辑 嵌入链接 使用完整模式 简单模式下只可以编辑链接,如需要设置图片标题或替代文本的话需要打开编辑图片页面点击下方的蓝字切换到完整模式,输入替代文本和标题后点击嵌入图像即可。 另外,点击图片后显示的悬浮窗内也可以修改图片的对齐方式(比如居中对齐) 除了图片以外,选中文字后同样也会出现悬浮框可以修改文字样式。 生成静态网页 在完成编写之后,就可以生成网页了。 运行以下命令: hugo 或携带一些参数: hugo --theme=stack --baseUrl="https://blog.reincarnatey.net/" 随后可以看到根目录下出现了 public 文件夹,该文件夹下就是生成的静态网页。你可以将该文件夹内的文件上传到服务器,或是 push 到你的 github 仓库部署到你使用的托管服务,访问你的域名就可以看到了。 使用 git 一键推送 如果你需要 push 到 github 仓库,那么在配置完成 git 后你可以使用这段代码进行初始化: cd public git init git remote add origin 你的仓库地址.git git add -A git commit -m "create" git push -u origin master 初始化后,以后你可以使用此脚本进行一键 push: push.bat cd public git add -A git commit -m "update" git push -u origin master 当然,你也可以将生产静态网页的命令加入到一键 push 的脚本的开头,可以实现一键生成并推送,这里就不作过多演示了。 至此,一个崭新的博客已然搭建完成。但搭建只是第一步,后续还需要更新和维护,按照以上流程编写、生成、部署即可。 附录 参考文献 Hugo中文文档 Hugo:Stack主题 - 建站指南 Stack官方文档 GitHub - Mantyke/Hugo-stack-theme-mod GitHub - chinayangxiaowei/marktext-chinese-language-pack: Simplified Chinese language pack for marktext Waline官方文档 PicGo官方文档 Hugo 代码高亮 版权信息 本文原载于 reincarnatey.net,遵循 CC BY-NC-SA 4.0 协议,复制请保留原文出处。

2023/1/15
articleCard.readMore