zoukankan      html  css  js  c++  java
  • Codeforces Round #554 (Div. 2) F2. Neko Rules the Catniverse (Large Version) (矩阵快速幂 状压DP)

    题意

    nn个点,每个点只能走到编号在[1,min(n+m,1)][1,min(n+m,1)]范围内的点。求路径长度恰好为kk的简单路径(一个点最多走一次)数。
    1n109,1m4,1kmin(n,12)1le nle 10^9,1le mle 4,1le kle min(n,12)

    分析

    直接考虑走路径的话不能判有没有走过,然后就把路径转化为一个序列,每次往里面插入新的点(神了)。因为一个点可以走到比他小的所有点,那么我们把点从大到小插入。

    假设现在已有序列为p1,p2,p3,...,pkp_1,p_2,p_3,...,p_k。那么当前插入一个点ii

    假设插在pjp_jpj+1p_{j+1}之间,必须满足pjp_j能走到ii并且ii能走到pj+1p_{j+1}。由于ii是最小的,那么所有pjp_j都能走到ii,所以只用考虑ii能走到哪些点。

    • 一种情况是直接放在最后。
    • 另一种情况是i+mpj+1i+mge p_{j+1}。那么满足这个式子的pj+1p_{j+1}最多有m(4)m(le 4)个。那么就把[i+1,i+m][i+1,i+m]mm个数有没有出现在序列中过状压成mm22进制数,记为SS。当前方案就是bitcount(S)bitcount(S)SS22进制中有多少个11)。

    那么一共就有bitcount(S)+1bitcount(S)+1种方案。

    另外还可以不插入。

    所以DPDP状态设为f[i][j][S]f[i][j][S]表示到ii这个点,序列长度为jj,上述状态为SS的方案数,转移方程为:
    f[i+1][j+1][(S<<11)&(2m1)]+=f[i][j][S](bitcount(S)+1)f[i+1][j][(S<<10)&(2m1)]+=f[i][j][S]egin{aligned} f[i+1][j+1][(S<<1|1)&(2^m-1)]&+=f[i][j][S]*(bitcount(S)+1)\ f[i+1][j][(S<<1|0)&(2^m-1)]&+=f[i][j][S]end{aligned}

    由于nn比较大,就矩阵加速就行了。这个矩阵快速幂还是挺好写的。。

    CODE

    #include <bits/stdc++.h>
    using namespace std;
    const int MAXN = 100005;
    const int mod = 1e9 + 7;
    int n, k, m, all;
    inline void add(int &x, int y) { x += y; if(x >= mod) x -= mod; }
    struct mat {
    	int a[210][210]; //最大状态数为(k+1)*(1<<m)<=(12+1)*(2^4)=208
    	mat() { memset(a, 0, sizeof a); }
    	inline mat operator *(const mat &o)const {
    		mat re;
    		for(int k = 0; k < all; ++k)
    			for(int i = 0; i < all; ++i) if(a[i][k])
    				for(int j = 0; j < all; ++j) if(o.a[k][j])
    					add(re.a[i][j], 1ll * a[i][k] * o.a[k][j] % mod);
    		return re;
    	}
    	inline mat operator ^(int b)const {
    		mat re, A = *this;
    		for(int i = 0; i < all; ++i) re.a[i][i] = 1;
    		while(b) {
    			if(b & 1) re = re * A;
    			A = A * A; b >>= 1;
    		}
    		return re;
    	}
    };
    inline int enc(int K, int S) { return K*(1<<m) + S; }
    inline int nxt(int S, bool x) { return ((S<<1)|x) & ((1<<m)-1); }
    int cnt[16];
    int main () {
    	scanf("%d%d%d", &n, &k, &m);
    	all = (k+1)*(1<<m); //所有状态数
    	mat trans, ans;
    	ans.a[0][enc(0, 0)] = 1;
    	for(int s = 1; s < (1<<m); ++s) cnt[s] = cnt[s>>1] + (s&1); //预处理2进制下有多少个1
    	for(int i = 0; i <= k; ++i)
    		for(int s = 0; s < (1<<m); ++s) {
    			if(i < k) trans.a[enc(i, s)][enc(i+1, nxt(s, 1))] = cnt[s]+1;
    			trans.a[enc(i, s)][enc(i, nxt(s, 0))] = 1;
    		}
    	ans = ans * (trans ^ n);
    	int Ans = 0;
    	for(int s = 0; s < (1<<m); ++s)
    		add(Ans, ans.a[0][enc(k, s)]);
    	printf("%d
    ", (Ans + mod) % mod);
    }
    
  • 相关阅读:
    flex+java+blazeds 多通道好文
    如何保持PC客户端一直处于登录状态
    函数进阶
    数据类型扩展
    python编码规范
    xpath轴定位
    IDEA Terminal 运行mvn命令报该命令不是内部命令
    java环境安装Firefox驱动/IE驱动
    java环境添加chrome驱动
    java安装selenium webdriver环境
  • 原文地址:https://www.cnblogs.com/Orz-IE/p/12039278.html
Copyright © 2011-2022 走看看