zoukankan      html  css  js  c++  java
  • 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(mlog ?)$ 

    代码

    #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
    ",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;
    }
    

      

  • 相关阅读:
    java:transient是什么,有什么作用
    如何阅读java源码
    java里面list是引用的好例子
    sort给文件按照大小排序
    HBase的rowkey排序和scan输出顺序
    记录一次事故——idea,sbt,scala
    一个简单的synchronized多线程问题、梳理与思考
    Android TextView文字描边的实现!!
    android中include标签的使用
    layout_weight 的解释及使用
  • 原文地址:https://www.cnblogs.com/zhouzhendong/p/UOJ440.html
Copyright © 2011-2022 走看看