UOJ#440. 【NOIP2018】填数游戏 动态规划
原文链接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 ?)$
代码
- #include <bits/stdc++.h>
- #define clr(x) memset(x,0,sizeof (x))
- #define For(i,a,b) for (int i=a;i<=b;i++)
- #define Fod(i,b,a) for (int i=b;i>=a;i--)
- #define pb(x) push_back(x)
- #define mp(x,y) make_pair(x,y)
- #define fi first
- #define se second
- #define _SEED_ ('C'+'L'+'Y'+'A'+'K'+'I'+'O'+'I')
- #define outval(x) printf(#x" = %d\n",x)
- #define outvec(x) printf("vec "#x" = ");for (auto _v : x)printf("%d ",_v);puts("")
- #define outtag(x) puts("----------"#x"----------")
- #define outarr(a,L,R) printf(#a"[%d...%d] = ",L,R);\
- For(_v2,L,R)printf("%d ",a[_v2]);puts("");
- using namespace std;
- typedef long long LL;
- typedef unsigned long long ULL;
- typedef vector <int> vi;
- LL read(){
- LL x=0,f=0;
- char ch=getchar();
- while (!isdigit(ch))
- f|=ch=='-',ch=getchar();
- while (isdigit(ch))
- x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
- return f?-x:x;
- }
- const int mod=1e9+7;
- int Pow(int x,int y){
- int ans=1;
- for (;y;y>>=1,x=(LL)x*x%mod)
- if (y&1)
- ans=(LL)ans*x%mod;
- return ans;
- }
- void Add(int &x,int y){
- if ((x+=y)>=mod)
- x-=mod;
- }
- void Del(int &x,int y){
- if ((x-=y)<0)
- x+=mod;
- }
- int Add(int x){
- return x>=mod?x-mod:x;
- }
- int Del(int x){
- return x<0?x+mod:x;
- }
- int n,m;
- map <int,int> f,g;
- map <int,int> :: iterator it;
- int Log[257];
- int calc(int s1,int s2){
- int a=(s1&s2)|(~s1&~s2);
- return a?Log[a&-a]:n-1;
- }
- int Hash(int l,int a,int b){
- return l<<(n+5)|a<<(n+1)|b;
- }
- void upd(int l,int a,int b,int v){
- int s=(b>>1)&((1<<l)-1),r=((b>>1)>>l)<<l;
- for (int t=s;;t=(t-1)&s){
- int nxl=min(l,calc(s,t)+1);
- Add(g[Hash(nxl,a-1,t|r)],v);
- if (!t)
- break;
- }
- }
- int main(){
- n=read(),m=read();
- For(i,2,256)
- Log[i]=Log[i>>1]+1;
- int ub=(1<<n)-1;
- For(i,0,ub)
- f[Hash(n,n,i)]=1;
- For(i,2,m){
- g.clear();
- for (it=f.begin();it!=f.end();it++){
- int now=(*it).fi,l=now>>(n+5),a=now>>(n+1)&15,b=now&ub;
- if (a>l)
- upd(l,a,b,(*it).se);
- else {
- if (a!=n)
- upd(l,a+1,b,(*it).se);
- upd(l,a+1,b|(1<<a),(*it).se);
- }
- }
- swap(f,g);
- }
- int ans=0;
- for (it=f.begin();it!=f.end();it++){
- int v=(*it).se,t=n-((*it).fi>>(n+1)&15);
- Add(ans,(LL)v*(1<<t)%mod);
- }
- cout<<ans<<endl;
- return 0;
- }
UOJ#440. 【NOIP2018】填数游戏 动态规划的更多相关文章
- [Noip2018]填数游戏
传送门 Description 耳熟能详,就不多说了 Solution 对于一个不会推式子的蒟蒻,如何在考场优雅地通过此题 手玩样例,发现对于 \(n=1\) , \(ans=2^m\) .对于 \( ...
- NOIP2018 填数游戏 搜索、DP
LOJ 感觉这个题十分好玩于是诈尸更博.一年之前的做题心得只有这道题还记得清楚-- 设输入为\(n,m\)时的答案为\(f(n,m)\),首先\(f(n,m)=f(m,n)\)所以接下来默认\(n \ ...
- 【题解】NOIP2018 填数游戏
题目戳我 \(\text{Solution:}\) 题目标签是\(dp,\)但是纯暴力打表找规律可以有\(65\)分. 首先是对于\(O(2^{nm}*nm)\)的暴力搜索,显然都会. 考虑几条性质: ...
- 【比赛】NOIP2018 填数游戏
打表找规律.... #include<bits/stdc++.h> #define ui unsigned int #define ll long long #define db doub ...
- @NOIP2018 - D2T2@ 填数游戏
目录 @题目描述@ @题解@ @代码@ @题目描述@ 小 D 特别喜欢玩游戏.这一天,他在玩一款填数游戏. 这个填数游戏的棋盘是一个 n×m 的矩形表格.玩家需要在表格的每个格子中填入一个数字(数字 ...
- 【逆向笔记】2017年全国大学生信息安全竞赛 Reverse 填数游戏
2017年全国大学生信息安全竞赛 Reverse 填数游戏 起因是吾爱破解大手发的解题思路,觉得题挺有意思的,就找来学习学习 这是i春秋的下载链接 http://static2.ichunqiu.co ...
- luogu P5023 填数游戏
luogu loj 被这道题送退役了 题是挺有趣的,然而可能讨论比较麻烦,肝了2h 又自闭了,鉴于CSP在即,就只能先写个打表题解了 下面令\(n<m\),首先\(n=1\)时答案为\(2^m\ ...
- JZOJ5965【NOIP2018提高组D2T2】填数游戏
题目 作为NOIP2018的题目,我觉得不需要把题目贴出来了. 大意就是,在一个n∗mn*mn∗m的010101矩阵中,从左上角到右下角的路径中,对于任意的两条,上面的那条小于下面的那条.问满足这样的 ...
- NOIP2018 Day2T2 填数游戏
下面先给出大家都用的打表大法: 首先我们可以发现 \(n \le 3\) 的情况有 \(65pts\),而 \(n\) 这么小,打一下表何乐而不为呢?于是我写了一个爆枚每个位置再 \(check\) ...
随机推荐
- git合并常见冲突
如果一个文件在服务器上已经做了修改,然后在本地开发中又做了一些修改的时候,再发布这个文件时很容易造成代码冲突,错误如下, error: Your local changes to the follow ...
- SpringCloud笔记六:Hystrix
目录 Hystrix是什么? Hystrix服务熔断 新建Hystrix项目 修改yml Maven的pom.xml添加hystrix引用 修改Controller Hystrix服务降级 修改api ...
- Vue学习笔记一:初识Vue
目录 什么是Vue? 为什么要学习前端框架? MVC,MVP 和 MVVM 最简单的入门小案例 下载Vue.js 新建文件结构 写一个html 运行 可笑的小报错 Vue和MVVM 什么是Vue? V ...
- GIT-Linux(CentOS7)系统部署git服务器
GIT-Linux(CentOS7)系统部署git服务器 root账号登录 一. 安装并配置必要的依赖关系在CentOS系统上安装所需的依赖:ssh,防火墙,postfix(用于邮件通知) ,wget ...
- [源码分析]StringBuffer
[源码分析]StringBuffer StringBuffer是继承自AbstractStringBuilder的. 这里附上另外两篇文章的连接: AbstractStringBuilder : ht ...
- HBase海量数据存储
1.简介 HBase是一个基于HDFS的.分布式的.面向列的非关系型数据库. HBase的特点 1.海量数据存储,HBase表中的数据能够容纳上百亿行*上百万列. 2.面向列的存储,数据在表中是按照列 ...
- 2018-2019-2 《Java程序设计》第8周学习总结
20175319 2018-2019-2 <Java程序设计>第8周学习总结 教材学习内容总结 本周学习<Java程序设计>第十五章: 泛型: 泛型(Generics)的主要目 ...
- 【转】关于Tomcat下项目线程启动两次的问题
最近遇见了一个很搞得事情,在tomcat下启动项目时自己写的定时程序被执行了两次,导致程序启动了两个线程,使定时任务在几秒间隔内执行了两次,后来通过日志查到,原来是tomcat将项目启动了两次,为什么 ...
- tomcat设置为开机自启动
第一步:设置环境变量(在java环境变量配置完成的情况下) 计算机右键——>属性——>高级系统设置——>环境变量——>在用户变量中心新建CATALINA_HOME变量 编辑pa ...
- hadoop启动 datanode的live node为0
hadoop启动 datanode的live node为0 浏览器访问主节点50070端口,发现 Data Node 的 Live Node 为 0 查看子节点的日志 看到 可能是无法访问到主节点的9 ...