原文链接www.cnblogs.com/zhouzhendong/p/UOJ440.html

前言

菜鸡选手到省选了才做联赛题。

题解

首先我们分析一下性质:

1. 假如一个格子是 0,那么它的右上角一定是 0 。

2. 假如一个格子的左边和上面两个格子一样,那么从这个格子到终点的任何两条路径相同。

不难发现,对于第 3 个斜列,我们发现这个斜列至少有一对相邻的相同格子。

也就是说,从第 3 行第 3 列这个格子到达终点的所有路径都相同。

设 $dp[c][i][j][k]$ 表示前 $c$ 列,最后一列的第 $i+1$ 个格子到终点的所有路径相同,最后一列当前有 $j$ 个数,这个 $j$ 个数状压起来是 $k$ ,这种情况下的方案数。

由于之前发现的性质,我们可以发现这种DP状态到第 3 列开始之后就很少了,到第 8 列以后就稳定只有 8 个状态了。

所以大力转移即可。

时间复杂度 $O(m)$ 。

我偷懒用了Map,时间复杂度变成 $O(m\log ?)$

代码

  1. #include <bits/stdc++.h>
  2. #define clr(x) memset(x,0,sizeof (x))
  3. #define For(i,a,b) for (int i=a;i<=b;i++)
  4. #define Fod(i,b,a) for (int i=b;i>=a;i--)
  5. #define pb(x) push_back(x)
  6. #define mp(x,y) make_pair(x,y)
  7. #define fi first
  8. #define se second
  9. #define _SEED_ ('C'+'L'+'Y'+'A'+'K'+'I'+'O'+'I')
  10. #define outval(x) printf(#x" = %d\n",x)
  11. #define outvec(x) printf("vec "#x" = ");for (auto _v : x)printf("%d ",_v);puts("")
  12. #define outtag(x) puts("----------"#x"----------")
  13. #define outarr(a,L,R) printf(#a"[%d...%d] = ",L,R);\
  14. For(_v2,L,R)printf("%d ",a[_v2]);puts("");
  15. using namespace std;
  16. typedef long long LL;
  17. typedef unsigned long long ULL;
  18. typedef vector <int> vi;
  19. LL read(){
  20. LL x=0,f=0;
  21. char ch=getchar();
  22. while (!isdigit(ch))
  23. f|=ch=='-',ch=getchar();
  24. while (isdigit(ch))
  25. x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
  26. return f?-x:x;
  27. }
  28. const int mod=1e9+7;
  29. int Pow(int x,int y){
  30. int ans=1;
  31. for (;y;y>>=1,x=(LL)x*x%mod)
  32. if (y&1)
  33. ans=(LL)ans*x%mod;
  34. return ans;
  35. }
  36. void Add(int &x,int y){
  37. if ((x+=y)>=mod)
  38. x-=mod;
  39. }
  40. void Del(int &x,int y){
  41. if ((x-=y)<0)
  42. x+=mod;
  43. }
  44. int Add(int x){
  45. return x>=mod?x-mod:x;
  46. }
  47. int Del(int x){
  48. return x<0?x+mod:x;
  49. }
  50. int n,m;
  51. map <int,int> f,g;
  52. map <int,int> :: iterator it;
  53. int Log[257];
  54. int calc(int s1,int s2){
  55. int a=(s1&s2)|(~s1&~s2);
  56. return a?Log[a&-a]:n-1;
  57. }
  58. int Hash(int l,int a,int b){
  59. return l<<(n+5)|a<<(n+1)|b;
  60. }
  61. void upd(int l,int a,int b,int v){
  62. int s=(b>>1)&((1<<l)-1),r=((b>>1)>>l)<<l;
  63. for (int t=s;;t=(t-1)&s){
  64. int nxl=min(l,calc(s,t)+1);
  65. Add(g[Hash(nxl,a-1,t|r)],v);
  66. if (!t)
  67. break;
  68. }
  69. }
  70. int main(){
  71. n=read(),m=read();
  72. For(i,2,256)
  73. Log[i]=Log[i>>1]+1;
  74. int ub=(1<<n)-1;
  75. For(i,0,ub)
  76. f[Hash(n,n,i)]=1;
  77. For(i,2,m){
  78. g.clear();
  79. for (it=f.begin();it!=f.end();it++){
  80. int now=(*it).fi,l=now>>(n+5),a=now>>(n+1)&15,b=now&ub;
  81. if (a>l)
  82. upd(l,a,b,(*it).se);
  83. else {
  84. if (a!=n)
  85. upd(l,a+1,b,(*it).se);
  86. upd(l,a+1,b|(1<<a),(*it).se);
  87. }
  88. }
  89. swap(f,g);
  90. }
  91. int ans=0;
  92. for (it=f.begin();it!=f.end();it++){
  93. int v=(*it).se,t=n-((*it).fi>>(n+1)&15);
  94. Add(ans,(LL)v*(1<<t)%mod);
  95. }
  96. cout<<ans<<endl;
  97. return 0;
  98. }

  

UOJ#440. 【NOIP2018】填数游戏 动态规划的更多相关文章

  1. [Noip2018]填数游戏

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

  2. NOIP2018 填数游戏 搜索、DP

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

  3. 【题解】NOIP2018 填数游戏

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

  4. 【比赛】NOIP2018 填数游戏

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

  5. @NOIP2018 - D2T2@ 填数游戏

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

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

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

  7. luogu P5023 填数游戏

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

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

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

  9. NOIP2018 Day2T2 填数游戏

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

随机推荐

  1. git合并常见冲突

    如果一个文件在服务器上已经做了修改,然后在本地开发中又做了一些修改的时候,再发布这个文件时很容易造成代码冲突,错误如下, error: Your local changes to the follow ...

  2. SpringCloud笔记六:Hystrix

    目录 Hystrix是什么? Hystrix服务熔断 新建Hystrix项目 修改yml Maven的pom.xml添加hystrix引用 修改Controller Hystrix服务降级 修改api ...

  3. Vue学习笔记一:初识Vue

    目录 什么是Vue? 为什么要学习前端框架? MVC,MVP 和 MVVM 最简单的入门小案例 下载Vue.js 新建文件结构 写一个html 运行 可笑的小报错 Vue和MVVM 什么是Vue? V ...

  4. GIT-Linux(CentOS7)系统部署git服务器

    GIT-Linux(CentOS7)系统部署git服务器 root账号登录 一. 安装并配置必要的依赖关系在CentOS系统上安装所需的依赖:ssh,防火墙,postfix(用于邮件通知) ,wget ...

  5. [源码分析]StringBuffer

    [源码分析]StringBuffer StringBuffer是继承自AbstractStringBuilder的. 这里附上另外两篇文章的连接: AbstractStringBuilder : ht ...

  6. HBase海量数据存储

    1.简介 HBase是一个基于HDFS的.分布式的.面向列的非关系型数据库. HBase的特点 1.海量数据存储,HBase表中的数据能够容纳上百亿行*上百万列. 2.面向列的存储,数据在表中是按照列 ...

  7. 2018-2019-2 《Java程序设计》第8周学习总结

    20175319 2018-2019-2 <Java程序设计>第8周学习总结 教材学习内容总结 本周学习<Java程序设计>第十五章: 泛型: 泛型(Generics)的主要目 ...

  8. 【转】关于Tomcat下项目线程启动两次的问题

    最近遇见了一个很搞得事情,在tomcat下启动项目时自己写的定时程序被执行了两次,导致程序启动了两个线程,使定时任务在几秒间隔内执行了两次,后来通过日志查到,原来是tomcat将项目启动了两次,为什么 ...

  9. tomcat设置为开机自启动

    第一步:设置环境变量(在java环境变量配置完成的情况下) 计算机右键——>属性——>高级系统设置——>环境变量——>在用户变量中心新建CATALINA_HOME变量 编辑pa ...

  10. hadoop启动 datanode的live node为0

    hadoop启动 datanode的live node为0 浏览器访问主节点50070端口,发现 Data Node 的 Live Node 为 0 查看子节点的日志 看到 可能是无法访问到主节点的9 ...