\(FWT\)--快速沃尔什变化学习笔记

知识点

  \(FWT\)就是求两个多项式的位运算卷积。类比\(FFT\),\(FFT\)大多数求的卷积形式为\(c_n=\sum\limits_{i+j=n}a_i*b_j\)的形式。而\(FWT\)则求的卷积形式为\(c_n=\sum\limits_{i\oplus j=n}\),如何做这个玩意呢,我们还是考虑分治。把它分成两个部分,一个部分是\(A_0\),一部分是\(A_1\),分别表示的是最高位为\(0/1\),然后对于与卷积来说\(f(A)=(f(A_0+A_1),f(A_1))\),对于或卷积来说\(f(A)=(f(A_0),f(A_0+A_1))\),对于异或卷积来说\(f(A)=(f(A_0+A_1),f(A_0-A_1))\)。不会证明。。这是比较重要的几点,然后这些都是正变化。而逆变化对于与和或来说就是把\(+\)换成\(-\),但对于异或卷积来说,\(f(A)=(f((A_0+A_1)/2,f((A_0-A_1)/2)\)。知道了这些就可以开始做题了。

例题

\(1\),洛谷\(4717\)

  (https://www.luogu.org/problemnew/show/P4717)

  解题思路:

    对,就是模板题!

  \(Code\)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm> using namespace std;
typedef long long LL;
const int MAXN = 1<<18;
const int MOD = 998244353;
const int INV = 499122177; int limit,f[MAXN],g[MAXN],a[MAXN],b[MAXN],n; inline int rd(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {f=ch=='-'?0:1;ch=getchar();}
while(isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return f?x:-x;
} inline int add(int x,int y){
int ret=x+y;while(ret>=MOD) ret-=MOD;while(ret<0) ret+=MOD;return ret;
} void out(int x){
if(!x) return;out(x/10);putchar('0'+x%10);
} inline void OUT(int x){
if(x<0) putchar('-');
if(x==0) putchar('0');
else out(x);putchar(' ');
} inline void fwt1(int *f,int type){ //or
for(int mid=1;mid<limit;mid<<=1)
for(int j=0,len=mid<<1;j<limit;j+=len)
for(int k=j;k<mid+j;k++)
f[mid+k]=add(f[mid+k],type*f[k]);
} inline void fwt2(int *f,int type){ //and
for(int mid=1;mid<limit;mid<<=1)
for(int j=0,len=mid<<1;j<limit;j+=len)
for(int k=j;k<mid+j;k++)
f[k]=add(f[k],type*f[k+mid]);
} inline void fwt3(int *f,int type){ //xor
int x,y;
for(int mid=1;mid<limit;mid<<=1)
for(int j=0,len=mid<<1;j<limit;j+=len)
for(int k=j;k<mid+j;k++){
x=f[k];y=f[k+mid];
f[k]=add(x,y);f[k+mid]=add(x,-y);
if(type==-1) f[k]=(LL)f[k]*INV%MOD,f[k+mid]=(LL)f[k+mid]*INV%MOD;
}
} int main(){
n=rd();limit=1<<n;
for(int i=0;i<limit;i++) a[i]=f[i]=rd();
for(int i=0;i<limit;i++) b[i]=g[i]=rd();
fwt1(f,1);fwt1(g,1);
for(int i=0;i<limit;i++) f[i]=(LL)f[i]*g[i]%MOD;
fwt1(f,-1);
for(int i=0;i<limit;i++) OUT(f[i]);putchar('\n');
for(int i=0;i<limit;i++) f[i]=a[i];
for(int i=0;i<limit;i++) g[i]=b[i];
fwt2(f,1);fwt2(g,1);
for(int i=0;i<limit;i++) f[i]=(LL)f[i]*g[i]%MOD;
fwt2(f,-1);
for(int i=0;i<limit;i++) OUT(f[i]);putchar('\n');
for(int i=0;i<limit;i++) f[i]=a[i];
for(int i=0;i<limit;i++) g[i]=b[i];
fwt3(f,1);fwt3(g,1);
for(int i=0;i<limit;i++) f[i]=(LL)f[i]*g[i]%MOD;
fwt3(f,-1);
for(int i=0;i<limit;i++) OUT(f[i]);
return 0;
}

\(2\),\(bzoj\) \(4589\)

  (https://www.lydsy.com/JudgeOnline/problem.php?id=4589)

  解题思路:

  博弈论\(+ FWT\),首先\(nim\)游戏所有石子堆的异或和为\(0\)时先手必败。设\(f_i\)表示异或和为\(i\)时的情况,我们最后要求的是\(f_0\)。设\(a_i\)表示这堆石子能否为\(i\),那么对于两堆石子来说\(f_i=\sum\limits_{j\land k=i}a_j*a_k\),对于三堆石子来说\(f_i'=\sum\limits_{j\land k=i}f_j*a_k\)。这样就能发现规律了。首先我们先用\(FWT\)正变化一下,然后求一个\(pow(a_i,n)\),在逆变化回去。

  \(Code\)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#define int long long using namespace std;
const int MAXN = 50005;
const int MOD = 1e9+7;
const int INV = 500000004;
typedef long long LL; int n,m,prime[MAXN],cnt,limit=1,now;
LL a[MAXN<<1];
bool vis[MAXN]; inline LL add(LL ret){
while(ret>=MOD) ret-=MOD;while(ret<0) ret+=MOD;return ret;
} inline void fwt(LL *f,int type){
LL x,y;
for(int mid=1;mid<limit;mid<<=1)
for(int j=0,len=mid<<1;j<limit;j+=len)
for(int k=j;k<j+mid;k++){
x=f[k];y=f[k+mid];
f[k]=add(x+y);f[k+mid]=add(x-y);
if(type==-1) f[k]=f[k]*INV%MOD,f[k+mid]=f[k+mid]*INV%MOD;
}
} inline LL fast_pow(LL x,int y){
LL ret=1;
for(;y;y>>=1){
if(y&1) ret=ret*x%MOD;
x=x*x%MOD;
}
return ret;
} signed main(){
for(int i=2;i<=50000;i++){
if(!vis[i]) prime[++cnt]=i;
for(int j=1;j<=cnt && prime[j]*i<=50000;j++)
vis[i*prime[j]]=1;
}
while(~scanf("%lld%lld",&n,&m)){
if(n==1) {puts("0");continue;}
memset(a,0,sizeof(a));limit=1;now=0;
for(int i=1;i<=cnt;i++){
if(prime[i]<=m) a[prime[i]]=1;
else {now=prime[i-1];break;}
}if(!now) now=prime[cnt];
while(limit<=now) limit<<=1;
fwt(a,1);
for(int i=0;i<limit;i++) a[i]=fast_pow(a[i],n);
fwt(a,-1);printf("%lld\n",a[0]);
}
return 0;
}

\(3\),\(cf\) \(662C\)

  (https://www.luogu.org/problemnew/show/CF662C)

  解题思路:

  状态压缩\(+FWT\)。首先要把每一列压成一个\(0/1\)串,这样就变成一个长度为\(m\)的数列。然后可以\(2^n\)枚举一下翻哪一行,再\(O(m)\)的枚举一下列,让每一列先\(xor\)上外层枚举的状态,在取一个\(min(1,0)\)加入答案。这样时间复杂度是\(O(2^n*m)\)的,考虑优化。设\(ans_s\)为翻的行的状态为\(s\)时的最小值,\(f_i\)为刚开始\(i\)这个状态的个数,\(g_i\)表示\(i\)这个状态的\(0/1\)的最小值。那么\(ans_s=\sum\limits_{i\land j=s} f_i*g_j\),然后\(FWT\)一下即可。

  \(Code\)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm> using namespace std;
const int MAXN = 1<<21;
typedef long long LL; inline int rd(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {f=ch=='-'?0:1;ch=getchar();}
while(isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return f?x:-x;
} int n,m,S[100005];
LL f[MAXN],g[MAXN],ans=1e18; inline int min(int x,int y){
return x<y?x:y;
} inline void fwt(LL *f,int type){
LL x,y;
for(int mid=1;mid<(1<<n);mid<<=1)
for(int j=0,len=mid<<1;j<1<<n;j+=len)
for(int k=j;k<j+mid;k++){
x=f[k];y=f[k+mid];
f[k]=x+y;f[k+mid]=x-y;
if(type==-1) f[k]/=2,f[k+mid]/=2;
}
} int main(){
n=rd(),m=rd();char s[100005];
for(int i=1;i<=n;i++){
scanf("%s",s+1);
for(int j=1;j<=m;j++)
if(s[j]=='1') S[j]|=(1<<(i-1));
}
for(int i=1;i<=m;i++) f[S[i]]++;int cnt;
for(int i=0;i<(1<<n);i++){
cnt=__builtin_popcount(i);
g[i]=min(cnt,n-cnt);
}
fwt(f,1);fwt(g,1);
for(int i=0;i<(1<<n);i++) f[i]=f[i]*g[i];
fwt(f,-1);
for(int i=0;i<(1<<n);i++) ans=min(ans,f[i]);
printf("%lld\n",ans);
return 0;
}

[学习笔记] $FWT$的更多相关文章

  1. Log4j简单学习笔记

    log4j结构图: 结构图展现出了log4j的主结构.logger:表示记录器,即数据来源:appender:输出源,即输出方式(如:控制台.文件...)layout:输出布局 Logger机滤器:常 ...

  2. 小波说雨燕 第三季 构建 swift UI 之 UI组件集-视图集(六)Picker View视图 学习笔记

    想对PickerView进行操作,只能在代码中操作. 下面 ,再添加三个label组件,然后将所有组件配置到代码中(看代码),然后要实现对PickerView的操作,就要实现它的DataSource协 ...

  3. DSP bootloader学习笔记1

    DSP bootloader学习笔记1 彭会锋 参考: TMS320F28xx DSP中内部Flash的应用研究 http://wenku.baidu.com/view/83e9837931b765c ...

  4. [未完成]WebService学习第一天学习笔记

    [未完成]WebService学习第一天学习笔记[未完成]WebService学习第一天学习笔记

  5. 【转】FragmentTest学习笔记1

    原文网址:http://blog.csdn.net/hishentan/article/details/20734489 源码部分: BookContent.java package com.exam ...

  6. OSGi.NET 学习笔记

    OSGi.NET 学习笔记 [目录]   持续更新和调整中,本人学习笔记,非官方文档,难免疏漏,仅供参考. OSGi.NET SDK下载地址. 前言及环境准备 模块化和插件化 概念 实例 小结 面向服 ...

  7. uIP学习笔记

    uIP学习笔记 从零开始使用uIP freemodbus modbus TCP 学习笔记

  8. springcloud Eureka学习笔记

    最近在学习springcloud,抽空记录下学习笔记;主要记录Eureka的实现过程和高可用性的实现 Eureka是一个服务治理框架,它提供了Eureka Server和Eureka Client两个 ...

  9. jQgrid学习笔记

    jQgrid学习笔记

  10. 从今天开始 每天记录HTML,CSS 部分的学习笔记

    从今天开始 每天记录HTML,CSS 部分的学习笔记

随机推荐

  1. Manacher模板(O(n)内求最长回文串长度)

    转自:https://segmentfault.com/a/1190000008484167 /* 由于回文分为偶回文(比如 bccb)和奇回文(比如 bcacb),而在处理奇偶问题上会比较繁琐,所以 ...

  2. php封装支付

    <?php namespace Pay; use think\Config; vendor ('Pay.alipay.pagepay.service.AlipayTradeService'); ...

  3. CSP-S2019初赛游记

    考得不好,不过\(86.5\)分应该勉强能进. 比赛前 比赛前的一个星期是有点慌,因为初赛是必须要复习的.初赛和复赛很不一样,复赛可以得一等奖,初赛不一定能考得很好. 最恶心的当然是那些计算机的&qu ...

  4. Yii2的一些问题

    Yii2中删除能不能串着用 Yii2中find.findAll有什么区别 Yii2中User::findOne($id)和User::find->where(['id'=>1])-> ...

  5. [NOIP模拟测试7]visit 题解(组合数学+CRT+Lucas定理)

    Orz 因为有T的限制,所以不难搞出来一个$O(T^3)$的暴力dp 但我没试 据说有30分? 正解的话显然是组合数学啦 首先$n,m$可能为负,但这并没有影响, 我们可以都把它搞成正的 即都看作向右 ...

  6. codeforces 447E or 446C 线段树 + fib性质或二次剩余性质

    CF446C题意: 给你一个数列\(a_i\),有两种操作:区间求和:\(\sum_{i=l}^{r}(a[i]+=fib[i-l+1])\).\(fib\)是斐波那契数列. 思路 (一) codef ...

  7. C++——代码风格

    google代码风格 1.使用安全的分配器(allocator),如scoped_ptr,scoped_array 2.测试用的,其他的不能用: 2.1 友元 2.2 C++异常 2.3 RTTI 3 ...

  8. 问题:Error running 'lugia-web': Address loaclhost:1099 is already in use

    解决方法:cmd输入下面命令: 第一步: netstat -ano|findstr 1099 找到对应的pid 为3576.(每次不一样). 第二步:taskkill -f -pid 3576

  9. 小程序onclick的写法?

    bindtap="bindAction" 调用bindAction函数即可

  10. 解析TextView中的URL等指定特殊字符串与点击事件

    使用TextView时,有时可能需要给予TextView里的特定字符串,比如URL,数字特别的样式,必希望能够添加点击事件.比如发短信时,文字里的url就可以点击直接打开浏览器,数字可以点击拨打电话. ...