传送门

Description

耳熟能详,就不多说了

Solution

对于一个不会推式子的蒟蒻,如何在考场优雅地通过此题

  1. 手玩样例,发现对于 \(n=1\) , \(ans=2^m\) 。对于 \(n=2\) , \(ans=4\times 3^{m-1}\) 。或者干脆打出 \(n,m\le 3\) 的表

  2. 肉眼观察法,发现似乎有 \(f(n,m+1)=3f(n,m)\),但这并不是正确的,但如果你仅仅是这么认为了,你仍然能够获得很多分数

  3. 想结论,都是特别特别显然的那种:

  • \(f(n,m)=f(m,n)\) ,因而只要考虑 \(n\le m\) 的情况

  • 因为每步向右或向下,所以可按照步骤,把图分成一条条从左下到右上的斜线,对于单个格子进行考虑,发现每条斜线上的数单调不增(所以呢,你就可以通过枚举每条斜线在哪个位置开始变为\(0\)就可以了,简单打表)

  • 为了检验图的正确性,我们还需要发掘合法填数方式的更多性质:

    考虑怎样造成不合法,存在两条路径,它们在某个位置不合法了,那么它们之前路径对应的01串相同,且上一个位置相同,在不合法的这一步中,大的路径走了 \(0\) ,小的路径走了 \(1\) 。

    这启发我们,对于一张合法的图,如果某个点,存在两条到达它的路径对应相同的01串,那么它的后继相同,我们令这样的点叫\(A\)点,一个点是\(A\)点当且仅当它的前驱中有A点或者它的前驱的数相同

  1. 有了这么多的性质,我们发现其实可以打表拿很多分了,于是开始愉快地搜索,按照斜线一条条地搜,边搜边更新当前图的点中\(A\)类的点,同时,每条斜线上只有 \(0\) 和 \(1\) 的交界处可能导致不合法,判断一下它的上一个点是否是\(A\)类点就可以了
  2. 打出表了,发现结论 \(f(n,m+1)=3f(n,m),m>n\) !于是就很开心地过了
  3. 这个搜索是真的快,极限数据 \(n=8,m=9\) 都能在 \(0.6s\) 内过去,所以就连表都懒得打了,直接暴力就行了

Code 

  1. #include<bits/stdc++.h>
  2. #define ll long long
  3. #define db double
  4. #define reg register
  5. using namespace std;
  6. #define dbg1(x) cerr<<#x<<'='<<(x)<<' '
  7. #define dbg2(x) cerr<<#x<<'='<<(x)<<'\n'
  8. #define dbg3(x) cerr<<#x<<'\n'
  9. inline int read()
  10. {
  11. int x=0,f=1;char ch=getchar();
  12. while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
  13. while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
  14. return x*f;
  15. }
  16. const int P=1e9+7;
  17. int Add(int x,int y){return (x+y)%P;}
  18. int Mul(int x,int y){return (1ll*x*y)%P;}
  19. int fp(int x,int y){int r=1;if(y>0)for(;y;y>>=1,x=Mul(x,x))if(y&1)r=Mul(r,x);return r;}
  20. int n,m;ll ans=0;
  21. int Nm[20],X[20][20],Y[20][20];
  22. bool mk[10][10],mp[10][10];
  23. inline void getmk(int now)
  24. {
  25. int i,j;
  26. for(i=X[now][1],j=Y[now][1];i&&j<=m;--i,++j)
  27. if(i>1&&j>1)mk[i][j]=(mk[i-1][j]|mk[i][j-1]|(mp[i][j-1]==mp[i-1][j]));
  28. }
  29. void dfs(int now)
  30. {
  31. int i,j,p=Nm[now];getmk(now-1);
  32. for(i=0;i<=p;++i)
  33. {
  34. if(i)mp[X[now][i]][Y[now][i]]=true;
  35. if((i==0||i==p)||(i>0&&i<p&&!mk[X[now][i]-1][Y[now][i]]))
  36. if(now+1==n+m)++ans;else dfs(now+1);
  37. }
  38. for(i=1;i<=p;++i)mp[X[now][i]][Y[now][i]]=0;
  39. }
  40. int main()
  41. {
  42. freopen("game.in","r",stdin);
  43. freopen("game.out","w",stdout);
  44. n=read();m=read();
  45. if(n>m)swap(n,m);
  46. if(n==1)return 0*printf("%d\n",fp(2,m));
  47. int c=max(0,min(m-n,m-n-1));m-=c;
  48. reg int i,j;
  49. for(i=1;i<=n;++i)Nm[i]=i;
  50. for(i=n+1;i<m;++i)Nm[i]=n;
  51. for(i=m;i<n+m;++i)Nm[i]=n+m-i;
  52. for(i=1;i<=n;++i)X[i][1]=i,Y[i][1]=1;
  53. for(i=n+1;i<n+m;++i)X[i][1]=n,Y[i][1]=i-n+1;
  54. for(i=1;i<n+m;++i)for(j=2;j<=Nm[i];++j)X[i][j]=X[i][j-1]-1,Y[i][j]=Y[i][j-1]+1;
  55. dfs(1);printf("%lld\n",Mul(ans,fp(3,c)));
  56. }

Blog来自PaperCloud,未经允许,请勿转载,TKS!

[Noip2018]填数游戏的更多相关文章

  1. NOIP2018 填数游戏 搜索、DP

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

  2. 【题解】NOIP2018 填数游戏

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

  3. 【比赛】NOIP2018 填数游戏

    打表找规律.... #include<bits/stdc++.h> #define ui unsigned int #define ll long long #define db doub ...

  4. @NOIP2018 - D2T2@ 填数游戏

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

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

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

  6. luogu P5023 填数游戏

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

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

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

  8. NOIP2018 Day2T2 填数游戏

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

  9. UOJ#440. 【NOIP2018】填数游戏 动态规划

    原文链接www.cnblogs.com/zhouzhendong/p/UOJ440.html 前言 菜鸡选手到省选了才做联赛题. 题解 首先我们分析一下性质: 1. 假如一个格子是 0,那么它的右上角 ...

随机推荐

  1. jquery库与其他库(比如prototype)冲突的解决方法

    前端开发很容易会遇到jQuery库与其他库冲突的场景,比如和prototype库冲突. 实际上这种冲突是因为不同的第三方JS库争夺对$标识符的控制权引起的. 解决方法,就是使用jQuery提供的jQu ...

  2. log4j打印抛出异常时堆栈内容

    JAVA 中为快速定位,抛出异常的位置.往往需要在日志中描述. log.err("异常出错点", e); 会将出错时的堆栈信息显示出来. package com.jqyc.jase ...

  3. 美国 | USA B1/B2 十年签证到手记

    旅行生活美国 周一早上人还没到公司,就收到EMS快递员的电话,说有两份我的美国领事馆的签证快递,在前台要我签收一下. 美领馆的出签效率果然是高,上周三(8月10日)早上面签的,这三个工作日就把护照送到 ...

  4. Gordon家族(二)

    本文是 Gordon家族(一) 的续集. 16. GoLearn 介绍:Gordon博士为Go开发者提供了一系列机器学习的库,开箱即用. 地址:https://github.com/sjwhitwor ...

  5. [NOI2019] 弹跳

    题意: 给你平面上的$n$个点,共有$m$个弹跳装置. 每个弹跳装置可以从点$p_{i}$以$t_{i}$的代价跳到矩形$(L_{i},D_{i}),(R_{i},U_{i})$中的任何一个点. 现在 ...

  6. C# 用Singleton类构建多线程单例模式

    public sealed class Singleton    {        private static volatile Singleton uniqueInstance;        p ...

  7. ASP.NET MVC中Log4Net记录错误日志的使用

    第一.在管理NuGet程序包 =>下载 Log4Net 第二.在web.config配置Log4Net 1:在<configuration>节点下 <configSection ...

  8. 代码语法高亮踩坑-原理,问题, PRE元素及htmlentity

    语法高亮库基础原理 在研究使用能够在web页面上代码语法高显的解决方案时,发现有很多现成的开源库.比较中意的有prism.js,highlightjs.他们的原理基本上核心就两点: 1. 利用html ...

  9. linux 搭建swoole

    1安装依赖包yum -y install gcc gcc-c++ autoconf automake yum -y install zlib zlib-devel openssl openssl-de ...

  10. html书写行内元素时-tab和换行会在行内元素间引入间距

    目录 html文本中的控制字符会被解析为文本节点 书写行内元素时,换行符LF与水平制表符HT会引入莫名的元素间间隔 其他控制字符是否会引入间距的验证 html文本中的控制字符会被解析为文本节点 举例: ...