题目链接

这道题讲道理还是不错的,因为你需要不断挖掘其中的性质来帮助解题。可惜数据范围开在这里让考试时的我很慌,勉强也就写了$65$分就没了。回忆在考场上,思路是没有错的,就是发掘不够深入,思路还不够清晰。事实上考场上没有选择继续做这道题是对的,因为就算是我考后仔细分析之后,写完这道题仍然花了我不少时间。

我们可以循着思路一步步分析,一步步得到每一个性质。

题目中对其走过路径的字典序的比较提示我们按斜行分析。稍加思考我们就能得到一个明显的结论,就是对于某一个格子如果它是$1$,那它的右上角的那个格子就不能是$0$,这几乎就是题目条件的定义,因为有一条路径走到了这个格子,它就会在分叉的时候出问题。我们它这个性质总结一下就能得到我们需要的第一个结论:

1.对于每一个斜行,其$0/1$状态一定是存在一个分界点,使得其左下方都是$1$,其右上方都是$0$。

有以上的结论将大大减少每一个斜行的可行的$0/1$状态。我们接着思考,一对不合法的路径的出现,除了上述的情况,都可以归结为两条不同的路径以相同的$0/1$串走到了某一个格子,但是这个格子右边下边的两个格子的$0/1$是不同的,这同样会让矛盾出现。或许你会想这两条路径在$0/1$字典序出现分歧的时候并不一定在同一个格子里,但如果存在这种情况,那我们一定能找到前者所说的更加简单的情况。我们将形式地描述这个问题,我们称一个格子是“模糊”的,当且仅当存在两条不同的路径以相同的$0/1$串走到了这个格子。我们所发现的可以表述成:

2.如果某一个格子它左边上边的两个格子的$0/1$是相同的,或者它左边或上边有格子是模糊点,那这个格子就是模糊点;模糊点右边下边的两个格子的$0/1$必须相同。

这也是一个重要的结论,它为下一个结论的得到提供了一个有力的帮助。模糊点的传递性隐约让我们感觉到它们的排布不会错杂无序,事实上十分有规律。读者可以仔细推敲,利用归纳法简单证明以下结论:

3.去除第一行和第一列的格子后,每一条斜行最多只有一个格子是非模糊的,并且这个非模糊点一定在第二行或第二列。

这个性质令人惊讶,但它是真实的,并且不难证明。有了这个结论后,我们就可以有一个大致的想法,我们可以枚举整张图的模糊状态,状态数是$O(m)$的,因为斜行上一旦全是模糊点,接下来也一定都是模糊点。我们考虑对于一个给定的模糊状态,我们怎么去计算有多少$0/1$的填放方式满足整张图的模糊状态。假设我们枚举那个仅存的非模糊点最后出现在哪一个斜行,手模一下可以发现,为了保证这个非模糊点没有消失,前面的大多数斜行的$0/1$状态是唯一的,只有最开头的两斜行会有多种状态,并且为了让这个非模糊点在下一斜行中消失,这行和下一行的可行状态数也可以知道,那么算到这里方案数还是一个已知的常数。在非模糊点消失后,接下来每一斜行面临的决策都是一样的,对于后面的方案数只要快速幂即可。到这里为止,已经可以解决这道题了,利用此算法的复杂度是$O(mlogm)$。

讲到这里算法大致结束了。对于上述算法而言,我们枚举了非模糊点最后出现的位置然后算方案数,其算式的形式是相同的,我们可以把其中的式子化简一下,就能用等比数列求和直接算了,在此处当$n=m$的时候要求有$3$的逆元。所以,总复杂度为$O(logm)$。这道题的细节相当复杂,其中的有许多常数要手动算出来,也有一些角落需要特判,就算大致知道怎么做了之后实现起来也是不容易的。

这份代码是$O(mlogm)$的实现,写的时候也是有点逻辑顺序在里面的,总的来说是按照斜行的从上到下。可能其中有需要解释的地方,$C$函数用于求在$x + 1$斜行后全部都是模糊点的方案数的计算,one more case中是一个算非模糊点在第二列最后两个位置时特判,dd line是非模糊点在第$n$斜行的特判,last case是非模糊点在第二行最后两个位置时的特判。

  1. #include <cstdio>
  2. #include <algorithm>
  3.  
  4. typedef long long LL;
  5.  
  6. const int MOD = (int)1e9 + ;
  7.  
  8. int n, m, ans, p2, p3;
  9.  
  10. int Pow(int x, int b) {
  11. int r = ;
  12. for (; b; b >>= , x = (LL)x * x % MOD) if (b & ) r = (LL)r * x % MOD;
  13. return r;
  14. }
  15. int C(int x) { // end by x, calc after
  16. if (x >= m) return Pow(, n + m - x - );
  17. if (x >= n) return (LL)Pow(, m - x) * p2 % MOD;
  18. return (LL)Pow(, n - x) * p3 % MOD * p2 % MOD;
  19. }
  20.  
  21. int main() {
  22. scanf("%d%d", &n, &m);
  23. if (n > m) std::swap(n, m);
  24. p2 = Pow(, n - );
  25. p3 = Pow(, m - n);
  26. if (n == ) {
  27. printf("%d\n", Pow(, m));
  28. return ;
  29. }
  30. if (n == ) {
  31. printf("%lld\n", 4LL * Pow(, m - ) % MOD);
  32. return ;
  33. }
  34. ans = (ans + 16LL * C()) % MOD; // on line 2
  35. ans = (ans + ( + (n != ) + (m > )) * 4LL * C()) % MOD; // on line 3
  36. for (int i = ; i < n; ++i) {
  37. ans = (ans + 80LL * C(i + )) % MOD;
  38. }
  39. // one more case
  40. if (n > ) {
  41. if (n < m) ans = (ans + 32LL * C(n + )) % MOD;
  42. else ans = (ans + 24LL * C(n + )) % MOD;
  43. }
  44. if (n < m) ans = (ans + 8LL * C(n + )) % MOD;
  45. else ans = (ans + 6LL * C(n + )) % MOD;
  46.  
  47. // dd line
  48. if (n < m && n != ) ans = (ans + 32LL * C(n + )) % MOD;
  49. for (int i = n + ; i < m; ++i) {
  50. ans = (ans + 24LL * C(i + )) % MOD;
  51. }
  52. // last case
  53. if (n > || m > ) {
  54. if (n < m) ans = (ans + 18LL * C(m + )) % MOD;
  55. else ans = (ans + 24LL * C(m + )) % MOD;
  56. }
  57. ans = (ans + 6LL * C(m + )) % MOD;
  58.  
  59. printf("%d\n", ans);
  60. return ;
  61. }

这份代码是$O(logm)$的实现,其中把一些项合并过了,故而稍变简洁,但是无法从中得到具体的含义的。

  1. #include <cstdio>
  2. #include <algorithm>
  3. using namespace std;
  4.  
  5. typedef long long LL;
  6.  
  7. const int MOD = (int)1e9 + ;
  8.  
  9. int n, m, ans, p2, p3;
  10.  
  11. int Pow(int x, int y) {
  12. int r = , b = (y + MOD - ) % (MOD - );
  13. for (; b; b >>= , x = (LL)x * x % MOD) if (b & ) r = (LL)r * x % MOD;
  14. return r;
  15. }
  16.  
  17. int main() {
  18. scanf("%d%d", &n, &m);
  19. if (n > m) swap(n, m);
  20. p2 = Pow(, n - );
  21. p3 = Pow(, m - n);
  22. if (n == ) ans = Pow(, m);
  23. if (n == ) ans = 4LL * Pow(, m - ) % MOD;
  24. if (n == ) ans = 112LL * p3 % MOD;
  25. if (n > ) {
  26. ans = (3LL * p2 + 21LL * Pow(, * n - ) % MOD * p3) % MOD;
  27. if (n == m) ans = (ans + 27LL * p2) % MOD;
  28. else ans = (ans + (24LL * p3 % MOD + ) * p2) % MOD;
  29. ans = (ans + 80LL * p2 % MOD * Pow(, m - n - ) % MOD * (Pow(, n - ) - )) % MOD;
  30. ans = (ans + 12LL * p2 % MOD * (Pow(, max(, m - n - )) - )) % MOD;
  31. }
  32. printf("%d\n", ans);
  33. return ;
  34. }

$\bigodot$总结:

对于这道题的我的做法,或许与大多数做法不一定相同,它并没有要求什么算法,甚至没有类可归,重要的是要细心耐心地思考与推导。

【NOIP 2018】填数游戏(思考与推导)的更多相关文章

  1. 【逆向笔记】2017年全国大学生信息安全竞赛 Reverse 填数游戏

    2017年全国大学生信息安全竞赛 Reverse 填数游戏 起因是吾爱破解大手发的解题思路,觉得题挺有意思的,就找来学习学习 这是i春秋的下载链接 http://static2.ichunqiu.co ...

  2. @NOIP2018 - D2T2@ 填数游戏

    目录 @题目描述@ @题解@ @代码@ @题目描述@ 小 D 特别喜欢玩游戏.这一天,他在玩一款填数游戏. 这个填数游戏的棋盘是一个 n×m 的矩形表格.玩家需要在表格的每个格子中填入一个数字(数字 ...

  3. [Noip2018]填数游戏

    传送门 Description 耳熟能详,就不多说了 Solution 对于一个不会推式子的蒟蒻,如何在考场优雅地通过此题 手玩样例,发现对于 \(n=1\) , \(ans=2^m\) .对于 \( ...

  4. NOIP2018 填数游戏 搜索、DP

    LOJ 感觉这个题十分好玩于是诈尸更博.一年之前的做题心得只有这道题还记得清楚-- 设输入为\(n,m\)时的答案为\(f(n,m)\),首先\(f(n,m)=f(m,n)\)所以接下来默认\(n \ ...

  5. luogu P5023 填数游戏

    luogu loj 被这道题送退役了 题是挺有趣的,然而可能讨论比较麻烦,肝了2h 又自闭了,鉴于CSP在即,就只能先写个打表题解了 下面令\(n<m\),首先\(n=1\)时答案为\(2^m\ ...

  6. 【题解】NOIP2018 填数游戏

    题目戳我 \(\text{Solution:}\) 题目标签是\(dp,\)但是纯暴力打表找规律可以有\(65\)分. 首先是对于\(O(2^{nm}*nm)\)的暴力搜索,显然都会. 考虑几条性质: ...

  7. 2018.10.14 NOIP训练 猜数游戏(决策单调性优化dp)

    传送门 一道神奇的dp题. 这题的决策单调性优化跟普通的不同. 首先发现这道题只跟r−lr-lr−l有关. 然后定义状态f[i][j]f[i][j]f[i][j]表示猜范围为[L,L+i−1][L,L ...

  8. JZOJ5965【NOIP2018提高组D2T2】填数游戏

    题目 作为NOIP2018的题目,我觉得不需要把题目贴出来了. 大意就是,在一个n∗mn*mn∗m的010101矩阵中,从左上角到右下角的路径中,对于任意的两条,上面的那条小于下面的那条.问满足这样的 ...

  9. NOIP2018 Day2T2 填数游戏

    下面先给出大家都用的打表大法: 首先我们可以发现 \(n \le 3\) 的情况有 \(65pts\),而 \(n\) 这么小,打一下表何乐而不为呢?于是我写了一个爆枚每个位置再 \(check\) ...

随机推荐

  1. go语言之行--文件操作、命令行参数、序列化与反序列化详解

    一.简介 文件操作对于我们来说也是非常常用的,在python中使用open函数来对文件进行操作,而在go语言中我们使用os.File对文件进行操作. 二.终端读写 操作终端句柄常量 os.Stdin: ...

  2. 汇编 OD 调式

    OD调试  命令栏指令 一.OD调试 重新开始:Ctrl+F2 转到地址:CTRL+G 断点切换: F2 断点窗口: Alt+B 运行 : F9 暂停 : F12 单步步过: F8 //遇到CAL ...

  3. 项目 - RM 部署上centos7 之后出现的一些问题和解决方法

    系统版本: [root@localhost logs]# cat /etc/redhat-release CentOS Linux release (Core) 获取方法来自:https://www. ...

  4. service手动实例化(new)导致类中的spring对象无法注入的问题解决

    下面说的这个画横线的可能是错误的,因为我之前用controller继承父类的注解对象的时候成功了,所以可能这次的唯一原因就是 不该把本该从ioc容器中拿出的对象通过new的方式实例化,至于继承注解对象 ...

  5. 微信小程序之分享或转发功能(自定义button样式)

    小程序页面内发起转发 通过给 button 组件设置属性open-type="share",可以在用户点击按钮后触发 Page.onShareAppMessage 事件,如果当前页 ...

  6. SQLServer数据库还原:无法在已有的mdf文件上还原文件

    如果提示无法在已有的mdf文件上还原文件,请修改如下位置

  7. 01.如何把.py文件打包成为exe,重点讲解pyinstaller的用法

    1.应用场景 1.1 故事背景 我自己用python写了一个小程序发给其他同事用,给他的就是一个.py文件,不过他觉得比较麻烦,还要安装环境,他问我有没有简单一点的方式,我给一个exe文件,他就不用安 ...

  8. linux之grep 基础

    第一章 -a    将binary文件以text文件的方式搜寻数据-c    只输出匹配行的计数,计算找到匹配的次数-I(大写i)    不区分大小写(只适合用于单字符)-h    查询多文件时不显示 ...

  9. Unity---Inspector面板自定义

    一. 参数自定义 一个含有成员的类Player using System.Collections; using System.Collections.Generic; using UnityEngin ...

  10. ELK日志方案--使用Filebeat收集日志并输出到Kafka

    1,Filebeat简介 Filebeat是一个使用Go语言实现的轻量型日志采集器.在微服务体系中他与微服务部署在一起收集微服务产生的日志并推送到ELK. 在我们的架构设计中Kafka负责微服务和EL ...