FWT小结

核心思想:把 a , b a,b a,b 化成 f w t ( a ) , f w t ( b ) fwt(a),fwt(b) fwt(a),fwt(b),相乘后再化为 a a a

化的过程用的是分治

所以和FFT其实一模一样

OR / AND 卷积

不需要什么技巧,暴力分治转移即可

每次分治下去,相当于位数减一

注意合并过程中我们是计算对应位的贡献

因为其它位的贡献我们在分治下去时已经计算了

后面区间其他数贡献到前面某个位置已经统计到后面的那个位置了
FWT小结_第1张图片

OR

FWT小结_第2张图片
在这里插入图片描述
考虑如何还原

对上面式子进行换元即可

在这里插入图片描述

void OR(int *f, int x) {
	for(k=1, o=2; o<=n; o<<=1, k<<=1)
		for(i=0; i<n; i+=o)
			for(j=0; j<k; ++j) 
				f[i+j+k]=(f[i+j+k]+f[i+j]*x)%mo; 
}

AND

反过来好像就行了

FWT小结_第3张图片

void AND(int *f, int x) {
	for(k=1, o=2; o<=n; o<<=1, k<<=1)
		for(i=0; i<n; i+=o)
			for(j=0; j<k; ++j) 
				f[i+j]=(f[i+j]+f[i+j+k]*x)%mo; 
}

XOR卷积

这个需要点技巧

首先考虑若有转移系数,必须满足以下条件:

在这里插入图片描述

我们可以用以下方法构造:

在这里插入图片描述

构造的原理:

异或前后 1 的个数的奇偶性不变

因此得证:

在这里插入图片描述

在这里插入图片描述

所以有:

FWT小结_第4张图片

我们把它写成这个形式:

FWT小结_第5张图片

然后考虑如何FWT和IFWT

思考分治过程中popcount的变化

FWT小结_第6张图片

因此合并的过程可以表示为:

在这里插入图片描述

换元一下可以得到其逆变换:

FWT小结_第7张图片

void XOR(int *f, int x) {
	for(k=1, o=2; o<=n; o<<=1, k<<=1) 
		for(i=0; i<n; i+=o) 
			for(j=0; j<k; ++j) {
				f[i+j]+=f[i+j+k]; 
				f[i+j+k]=f[i+j]-2*f[i+j+k]; 
				f[i+j]=f[i+j]*x%mo; f[i+j+k]=f[i+j+k]*x%mo; 
			}
}

板子

#include
using namespace std;
#define int long long
inline int read(){int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;
ch=getchar();}while(ch>='0'&&ch<='9'){x=(x<<1)+
(x<<3)+(ch^48);ch=getchar();}return x*f;}
#define Z(x) (x)*(x)
#define pb push_back
//#define M
#define mo 998244353
#define N 200010
int pw(int a, int b) {
	int ans=1; 
	while(b) {
		if(b&1) ans*=a; 
		a*=a; b>>=1; 
		ans%=mo; a%=mo; 
	}
	return ans; 
}
const int inv2=pw(2, mo-2); 
int n, m, i, j, k, T;
int a[N], b[N], A[N], B[N], o; 

void cp() {
	for(i=0; i<n; ++i) a[i]=A[i], b[i]=B[i]; 
}

void pr() {
	for(i=0; i<n; ++i) printf("%lld ", (a[i]%mo+mo)%mo); 
	printf("\n"); 
}

void mul() {
	for(i=0; i<n; ++i) a[i]=a[i]*b[i]%mo; 
}

void OR(int *f, int x) {
	for(k=1, o=2; o<=n; o<<=1, k<<=1)
		for(i=0; i<n; i+=o)
			for(j=0; j<k; ++j) 
				f[i+j+k]=(f[i+j+k]+f[i+j]*x)%mo; 
}

void AND(int *f, int x) {
	for(k=1, o=2; o<=n; o<<=1, k<<=1)
		for(i=0; i<n; i+=o)
			for(j=0; j<k; ++j) 
				f[i+j]=(f[i+j]+f[i+j+k]*x)%mo; 
}

void XOR(int *f, int x) {
	for(k=1, o=2; o<=n; o<<=1, k<<=1) 
		for(i=0; i<n; i+=o) 
			for(j=0; j<k; ++j) {
				f[i+j]+=f[i+j+k]; 
				f[i+j+k]=f[i+j]-2*f[i+j+k]; 
				f[i+j]=f[i+j]*x%mo; f[i+j+k]=f[i+j+k]*x%mo; 
			}
}

signed main()
{
//	freopen("in.txt", "r", stdin);
//	freopen("out.txt", "w", stdout);
//	srand(time(NULL));
//	T=read();
//	while(T--) {
//
//	}
	n=read(); n=(1<<n); 
	for(i=0; i<n; ++i) A[i]=read(); 
	for(i=0; i<n; ++i) B[i]=read(); 
	cp(); OR(a, 1); OR(b, 1); mul(); OR(a, -1); pr(); 
	cp(); AND(a, 1); AND(b, 1); mul(); AND(a, -1); pr(); 
	cp(); XOR(a, 1); XOR(b, 1); mul(); XOR(a, inv2); pr(); 
	return 0;
}

你可能感兴趣的:(FWT,分治,卷积)