自嗨测试赛2

任意模数多项式乘法逆

多项式F与其乘法逆G相乘后,除常数项为1外,其他项系数都为0

由此列出式子,移项后就很明显能O(n^2)求出G1~Gn了

后面其实就是个矩阵加速数列递推了,特判完前n项后,直接矩阵快速幂就完了

Code
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define ll long long
using namespace std;

const int maxk = 15;
int n, K, mod, f[maxk], g[maxk];

struct Mat {
	int a[maxk][maxk];
	Mat() { memset(a, 0, sizeof a); }
	void Init() {
		for(int i = 1;i <= K; ++i) a[i][i] = 1;
	}
	Mat operator * (const Mat &B) const {
		Mat ans;
		for(int i = 1;i <= K; ++i) {
			for(int j = 1;j <= K; ++j) {
				for(int k = 1;k <= K; ++k) {
					(ans.a[i][j] += (ll)a[i][k]*B.a[k][j]%mod)%=mod;
				}
			}
		}
		return ans;
	}
};

Mat fp(Mat a, int x) {
	Mat ans; ans.Init();
	for(;x > 0;x >>= 1, a = a*a)
		if(x&1) ans = ans*a;
	return ans;
}

int main() {
	freopen("poly.in","r",stdin);
	freopen("poly.out","w",stdout);
	scanf("%d%d%d", &n, &K, &mod);
	for(int i = 1;i <= K; ++i) scanf("%d", &f[i]);
	g[0] = 1;
	for(int i = 1;i <= K; ++i) {
		int now = 0;
		for(int j = 1;j <= i; ++j) (now -= (ll)f[j]*g[i-j]%mod)%=mod;
		g[i] = (now%mod+mod)%mod;
	}
	if(n <= K) return printf("%d\n", g[n]), 0;
	Mat F, G;
	for(int i = 1;i <= K; ++i) G.a[i][1] = g[K-i+1];
	for(int i = 1;i <= K; ++i) F.a[1][i] = mod-f[i];
	for(int i = 2;i <= K; ++i) F.a[i][i-1] = 1;
	G = fp(F, n-K)*G;
	printf("%d\n", G.a[1][1]);
	return 0;
}

明明的随机数

  • 先求出k1行k2列各行各列都不相同的情况,要用到n^2递推第一类斯特林数,斯特林反演

  • 然后乘以第二类斯特林数S2(n,k1)*S2(m,k2)就行了,这两个数可以O(klogn)求

对于第一步直接搬题解了

Code
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define ll long long
using namespace std;

const int mod = 998244353;
int n, m, c, k1, k2;
int s1[3005][3005], fac[3005], fy[3005], f[3005], g[3005];

int fp(int a, int x, int ans = 1) {
	for(;x > 0;x >>= 1, a = (ll)a*a%mod) if(x&1) ans = (ll)ans*a%mod;
	return ans;
}

int C(int x, int y) {
	return (x<y||y<0) ? 0 : (ll)fac[x]*fy[y]%mod*fy[x-y]%mod;
}

void Init() {
	s1[0][0] = fac[0] = fy[0] = 1;
	for(int i = 1;i <= 3000; ++i) for(int j = 1;j <= i; ++j) s1[i][j] = (s1[i-1][j-1]+(ll)s1[i-1][j]*(i-1)%mod)%mod;
	for(int i = 1;i <= 3000; ++i) fac[i] = (ll)fac[i-1]*i%mod;
	fy[3000] = fp(fac[3000], mod-2);
	for(int i = 3000;i > 1; --i) fy[i-1] = (ll)fy[i]*i%mod;
}

int S2(int n, int m) {
	int S = 0;
	for(int i = 0;i <= m; ++i) {
		if(i&1) (S += mod-(ll)C(m, i)*fp(m-i, n)%mod)%=mod;
		else (S += (ll)C(m, i)*fp(m-i, n)%mod)%=mod;
	}
	return (ll)S*fy[m]%mod;
}

int main() {
	freopen("random.in","r",stdin);
	freopen("random.out","w",stdout);
	Init();
	int cs; scanf("%d", &cs);
	while(cs --> 0){
		scanf("%d%d%d%d%d", &n, &m, &k1, &k2, &c);
		if(c == 1) { printf("%d\n", k1==1&&k2==1);continue; }
		if(k1 == k2 && k1 == 1) { printf("%d\n", c%mod);continue; }
		for(int i = 1;i <= k1; ++i) {
			int now = 1, lst = fp(c, i);
			for(int j = 1;j <= k2; ++j) now = (ll)now*(lst-j+1)%mod;
			f[i] = (now%mod+mod)%mod;
			g[i] = 0;
			for(int j = 1;j <= i; ++j) {
				if((i-j)&1) (g[i] += mod-(ll)s1[i][j]*f[j]%mod)%=mod;
				else (g[i] += (ll)s1[i][j]*f[j]%mod)%=mod;
			}
		}
		printf("%lld\n", (ll)g[k1]*S2(n, k1)%mod*S2(m, k2)%mod);
	}
	return 0;
}

凯爹博弈

又现学了点博弈论,SG函数,不过这道题先咕了

Code

posted @ 2021-03-11 15:38  liuzhaoxu  阅读(47)  评论(0编辑  收藏  举报