[学习笔记]FWT——快速沃尔什变换
解决涉及子集配凑的卷积问题
一、介绍
1.基本用法
FWT快速沃尔什变换学习笔记
就是解决一类问题:
$f[k]=\sum_{i\oplus j=k}a[i]*b[j]$
基本思想和FFT类似。
首先转化成为另一个多项式$FWT(A),FWT(B)$
使得:$FWT(A\oplus B)=FWT(A)×FWT(B)$
这里,$×$是按位乘。这个是$O(n)$的。
然后,再$IFWT$回去即可。
类似于,直接过马路不好走。先从左边走上一座天桥,再从天桥走过去,再到马路右侧走下天桥。
就变成了$O(nlogn)$
$FWT$虽然不是非常容易理解,但是比较容易记忆。
(虽然一定要理解)
类比$FFT$的写法,就可以比较轻松记忆。
就是分治压缩、合并、分治解压的过程。
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize("Ofast")
#include<bits/stdc++.h>
#define reg register int
#define il inline
#define numb (ch^'0')
using namespace std;
typedef long long ll;
il void rd(int &x){
char ch;x=;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
il void prin(int x){
if(x/) prin(x/);
putchar(x%+'');
}
namespace Miracle{
const int N=+;
const int mod=;
const int inv2=;
int a[N],b[N];
int c[N],d[N];
int e[N];
int n;
void _or(int *f,int op){
for(reg p=;p<=n;p<<=){
int len=p/;
for(reg k=;k<n;k+=p){
for(reg l=k;l<k+len;++l){
if(op==)f[l+len]=(f[l+len]+f[l])%mod;
else f[l+len]=(f[l+len]-f[l]+mod)%mod;
}
}
}
}
void _and(int *f,int op){
for(reg p=;p<=n;p<<=){
int len=p/;
for(reg k=;k<n;k+=p){
for(reg l=k;l<k+len;++l){
if(op==)f[l]=(f[l+len]+f[l])%mod;
else f[l]=(f[l]-f[l+len]+mod)%mod;
}
}
}
}
void _xor(int *f,int op){
for(reg p=;p<=n;p<<=){
int len=p/;
for(reg k=;k<n;k+=p){
for(reg l=k;l<k+len;++l){
int x=f[l],y=f[l+len];
if(op==){
f[l]=(x+y)%mod;
f[l+len]=(x-y+mod)%mod;
}
else{
f[l]=(ll)(x+y)*inv2%mod;
f[l+len]=(ll)(x-y+mod)%mod*inv2%mod;
}
}
}
}
}
int main(){
scanf("%d",&n);
n=(<<n);
for(reg i=;i<n;++i) rd(a[i]),c[i]=a[i];
for(reg i=;i<n;++i) rd(b[i]),d[i]=b[i]; _or(c,);_or(d,);
for(reg i=;i<n;++i) e[i]=(ll)c[i]*d[i]%mod;
_or(e,-);
for(reg i=;i<n;++i){
prin(e[i]);putchar(' ');c[i]=a[i],d[i]=b[i];
}putchar('\n'); _and(c,);_and(d,);
for(reg i=;i<n;++i) e[i]=(ll)c[i]*d[i]%mod;
_and(e,-);
for(reg i=;i<n;++i){
prin(e[i]);putchar(' ');c[i]=a[i],d[i]=b[i];
}putchar('\n'); _xor(c,);_xor(d,);
for(reg i=;i<n;++i) e[i]=(ll)c[i]*d[i]%mod;
_xor(e,-);
for(reg i=;i<n;++i){
prin(e[i]);putchar(' ');//c[i]=a[i],d[i]=b[i];
}putchar('\n');
return ;
} }
int main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2018/11/22 15:08:15
*/
模板
2.子集卷积
https://www.cnblogs.com/Dance-Of-Faith/p/8818211.html
多加一维i,强制记录涉及集合sz大小
外层枚举sz
O(2^n*n^2)
(推荐使用FWT,因为比FMT常数小)
二、例题
留坑
[FWT] UOJ #310. 【UNR #2】黎明前的巧克力
三、FFT、NTT、FWT的比较
留坑
没啥可比较的。处理思路一致。
就是运算符的问题吧。
四、FWT、FMT的比较
留坑
FMT好写,FWT的与或卷积的第一步可以取代FMT
upda:2019.4.17
FMT可以代替FWT的与或卷积。IFMT把+改成-即可
(xor暂时不知道具体含义,估计也可以代替?)
实际上
FMT很辣鸡
相比之下,FWT做的事情完全包含FMT,并且常数是FMT的1/2!
[WC2018]州区划分(这个题我人傻常数大,必须用FWT卡常才能过)
所以还是写FWT吧
[学习笔记]FWT——快速沃尔什变换的更多相关文章
- FWT快速沃尔什变换学习笔记
FWT快速沃尔什变换学习笔记 1.FWT用来干啥啊 回忆一下多项式的卷积\(C_k=\sum_{i+j=k}A_i*B_j\) 我们可以用\(FFT\)来做. 甚至在一些特殊情况下,我们\(C_k=\ ...
- 浅谈算法——FWT(快速沃尔什变换)
其实FWT我啥都不会,反正就是记一波结论,记住就好-- 具体证明的话,推荐博客:FWT快速沃尔什变换学习笔记 现有一些卷积,形如 \(C_k=\sum\limits_{i\lor j=k}A_i*B_ ...
- Django RF:学习笔记(8)——快速开始
Django RF:学习笔记(8)——快速开始 安装配置 1.使用Pip安装Django REST Framework: pip install djangorestframework 2.在Sett ...
- [学习笔记]NTT——快速数论变换
先要学会FFT[学习笔记]FFT——快速傅里叶变换 一.简介 FFT会爆精度.而且浮点数相乘常数比取模还大. 然后NTT横空出世了 虽然单位根是个好东西.但是,我们还有更好的东西 我们先选择一个模数, ...
- 【学习笔记】快速傅里叶变换(FFT)
[学习笔记]快速傅里叶变换 学习之前先看懂这个 浅谈范德蒙德(Vandermonde)方阵的逆矩阵的求法以及快速傅里叶变换(FFT)中IDFT的原理--gzy hhh开个玩笑. 讲一下\(FFT\) ...
- 知识点简单总结——FWT(快速沃尔什变换),FST(快速子集变换)
知识点简单总结--FWT(快速沃尔什变换),FST(快速子集变换) 闲话 博客园的markdown也太傻逼了吧. 快速沃尔什变换 位运算卷积 形如 $ f[ i ] = \sum\limits_{ j ...
- [学习笔记] $FWT$
\(FWT\)--快速沃尔什变化学习笔记 知识点 \(FWT\)就是求两个多项式的位运算卷积.类比\(FFT\),\(FFT\)大多数求的卷积形式为\(c_n=\sum\limits_{i+j=n}a ...
- FWT快速沃尔什变换——基于朴素数学原理的卷积算法
这是我的第一篇学习笔记,如有差错,请海涵... 目录 引子 卷积形式 算法流程 OR卷积 AND卷积 XOR卷积 模板 引子 首先,考虑这是兔子 数一数,会发现你有一只兔子,现在,我再给你一只兔子 再 ...
- 初学FWT(快速沃尔什变换) 一点心得
FWT能解决什么 有的时候我们会遇到要求一类卷积,如下: Ci=∑j⊕k=iAj∗Bk\large C_i=\sum_{j⊕k=i}A_j*B_kCi=j⊕k=i∑Aj∗Bk此处乘号为普通乘法 ...
随机推荐
- java 线程Thread.Sleep详解 Thread.Sleep(0)的作用(转载)
我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段时间.那么你有没有正确的理解这个函数的用法呢? 思考下面这两个问题: 1.假设现在是 2008-4-7 12:00:00.000,如果 ...
- java.util (Collection接口和Map接口)
1:Collection和Map接口的几个主要继承和实现类 1.1 Collection接口 Collection是最基本的集合接口,一个Collection代表一 ...
- linuxmint 搜狗输入法安装
1.下载搜狗输入法linux安装包 2.进入安装包目录终端键入 dpkg -i [软件包名字] 3.设置语言选项中选择fcitx 4.重启电脑
- day 7-21 pymysql模块
一.安装的两种方法 第一种: #安装 pip3 install pymysql 第二种: 二.链接,执行sql,关闭(游标) import pymysql user = input("use ...
- 获取打开页面时的当前时间(yyyy-MM-dd hh:mm:ss)
Date.prototype.Format = function (fmt) { var o = { "M+": this.getMonth() + 1, //月份 "d ...
- 剑指offer(6)
题目: 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转. 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素. 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转 ...
- 阿里巴巴 Java开发手册1.4.0
<阿里巴巴Java开发手册1.4.0>下载地址: 下载地址:https://102.alibaba.com/downloadFile.do?file=1528269849853/Java_ ...
- 动态渲染页面爬取(Python 网络爬虫) ---Selenium的使用
Selenium 的使用 Selenium 是一个自动化测试工具,利用它可以驱动浏览器执行特定的动作,如点击.下拉等操作,同时还可以获取浏览器当前呈现的页面的源代码,做到可见即可爬.对于一些JavaS ...
- “耐撕”团队 2016.03.31 站立会议
1. 时间: 19:30--19:50 共计20分钟. 2. 成员: Z 郑蕊 * 组长 (博客:http://www.cnblogs.com/zhengrui0452/), P 濮成林(博客:ht ...
- Python——Django-应用的models.py内容
一.数据的相关配置 #数据库的相关配置 DATABASE = { 'default':{ #连接的数据库类型 'ENGINE':'django.db.backends.sqlite3', #连接数据库 ...