zoukankan      html  css  js  c++  java
  • BZOJ 4589 Hard Nim(FWT加速DP)

    题目链接  Hard Nim

    设$f[i][j]$表示前$i$个数结束后异或和为$j$的方案数

    那么$f[i][j] = f[i-1][j$ $hat{}$ $k]$,满足$k$为不大于$m$的质数。

    这个$DP$太暴力了。让我们冷静分析。

    设不大于m的质数从小到大分别为$c_{1}$, $c_{2}$, ..., $c_{k}$

    $f[i][j] = ∑f[i-1][j$ $hat{}$ $c[k]]$, 我们令$g[c[i]]$为$1$,其余为$0$。

    $f[i][j] = ∑f[i-1][j$ $hat{}$ $k] * g[k]$

    我们发现后边其实就是一个异或卷积的形式。

    于是就可以$FWT$了。

    但是左边那一维还是巨大……

    我们可以发现这个多项式乘法是满足结合律的

    于是在做逆变换之前的那个乘法的时候直接快速幂即可。

    #include <bits/stdc++.h>
    
    using namespace std;
    
    #define rep(i, a, b)	for (int i(a); i <= (b); ++i)
    #define dec(i, a, b)	for (int i(a); i >= (b); --i)
    #define MP		make_pair
    #define fi		first
    #define se		second
    
    
    typedef long long LL;
    
    const LL mod = 1e9 + 7;
    const LL rev = (mod + 1) >> 1;
    
    int n, m;
    int l;
    int tot = 0;
    int p[1 << 20];
    LL a[1 << 20], b[1 << 20], g[1 << 20];
    
    void pre(){
    	rep(i, 2, 5e4){
    		if (!g[i]) p[++tot] = i;
    		for (int j = 1; j <= tot && i * p[j] <= 5e4; ++j){
    			g[i * p[j]] = 1;
    			if (i % p[j] == 0) break;
    		}
    	}
    }
    
    void FWT(LL *a, int n){
    	for (int d = 1; d < n; d <<= 1)
    		for (int m = d << 1, i = 0; i < n; i += m)
    			for (int j = 0; j < d; j++){
    				LL x = a[i + j], y = a[i + j + d];
    				a[i + j] = (x + y) % mod, a[i + j + d] = (x - y + mod) % mod;
    
    			}
    }
    
    void UFWT(LL *a, int n){
    	for (int d = 1; d < n; d <<= 1)
    		for (int m = d << 1, i = 0; i < n; i += m)
    			for (int j = 0; j < d; j++){
    				LL x = a[i + j], y = a[i + j + d];
    				a[i + j] = 1LL * (x + y) * rev % mod, a[i + j + d] = (1LL * (x - y) * rev % mod + mod) % mod;
    			}
    }
    
    void solve(LL *a, LL *b, int n, int p){
    	a[0] = 1;
    	FWT(a, n);
    	FWT(b, n);
    	while (p){
    		if (p & 1){
    			rep(i, 0, n - 1) (a[i] *= b[i]) %= mod;
    		}
    
    		rep(i, 0, n - 1) (b[i] *= b[i]) %= mod;
    		p >>= 1;
    	}
    
    	UFWT(a, n);
    }
    
    int main(){
    
    	pre();
    
    	while (~scanf("%d%d", &n, &m)){
    		for (l = 1; l <= m; l <<= 1){;}
    
    		memset(a, 0, sizeof a);
    		memset(b, 0, sizeof b);
    
    		for (int i = 1; i <= tot && p[i] <= m; ++i){
    			b[p[i]] = 1;
    		}
    
    		solve(a, b, l, n);
    		printf("%lld
    ", a[0]);
    	}
    
    
    	return 0;
    }
    

      

  • 相关阅读:
    使用VC++生成调试信息
    在Xp home上安装Rose 2003
    SkyDrive注册方法
    vsftpd同时使用系统用户和虚拟用户验证
    如何查看linux系统版本
    在RedHat AS中安装SVN
    Vnc & Gdm
    (转)如何:在设备上安装 SQL Server Compact 3.5
    java培训学习笔记一
    因为此版本的应用程序不支持其项目类型(.csproj),若要打开它,请使用支持此类型项
  • 原文地址:https://www.cnblogs.com/cxhscst2/p/8563751.html
Copyright © 2011-2022 走看看