首页 > 其他分享 >快速傅里叶变换 学习笔记

快速傅里叶变换 学习笔记

时间:2024-08-19 22:26:48浏览次数:17  
标签:FR 变换 res 笔记 傅里叶 FL 多项式 Date omega

快速傅里叶变换 学习笔记

简介:

卷积是形如 \(C_k=\sum_{i\oplus j==k} A_i*B_j\) 的式子,其中 \(\oplus\) 为表示某种运算。

而快速傅里叶变换(FFT)用于在 \(O(n\log n)\) 的时间复杂度内求加法卷积。

对于一个 \(k\) 次多项式,如果我们知道了它在 \(k+1\) 个点上的值,那么我们可以求出这个多项式。而两个多项式在同一个点上的值相乘的话,就可以得到 这两个多项式相乘后的多项式 在这个点上的值。FFT 的主要思路就是先将两个多项式在若干个点上的取值,然后相乘,再通过点值求出两个多项式相乘的结果。

单位根

先考虑求点值。我们肯定不能暴力选择点去代入求值,这样的时间复杂度是 \(O(n^2)\) 的,要找一些有特殊性质的点去代入。

于是我们在复数域找到了单位根,它是在复平面上的单位圆上的点,表示为 \(\omega_n^k\) ,它的意义是将单位圆分割成 \(n\) 份,按顺指针方向的第 \(k\) 个点所表示的值。两个复数相乘时,它们的辐角相加(从实轴的正半轴开始,按逆时针旋转),模长相乘(该点到原点的距离)。于是我们可以得到单位根的以下性质:

\[\omega_n^i*\omega_n^j=\omega_n^{i+j}\\ \omega_n^k=\omega_n^{k\%n}\\ \omega_{n*i}^{k*i}=\omega_n^k\\ \omega_n^{k+n/2}=-\omega_n^k(当 n 为偶数时) \]

对于最后一条,我们可以将 \(n/2\) 看做旋转半个圆周,所以就是取相反数。

对于一个 \(n\) 次多项式,我们看看将单位根代入会有什么性质:

先将多项式 \(F[x]\) 拆成两个部分: \(FL[x]=x^0+x^2+x^4...,FR[x]=x^1+x^3+x^5...\) 。然后就可以将多项式 \(F[x]\) 这样表示:

\[F[x]=FL[x^2]+x*FR[x^2] \]

再分别将 \(\omega_n^k\) 和 \(\omega_n^{k+n/2}\) 代入:

\[F[\omega_n^k]=FL[(\omega_n^k)^2]+\omega_n^k*FR[(\omega_n^k)^2]\\ =FL[\omega_{n/2}^k]+\omega_n^k*FR[\omega_{n/2}^k]\\ F[\omega_n^{k+n/2}]=FL[(\omega_n^{k+n/2})^2]+\omega_n^{k+n/2}*FR[(\omega_n^{k+n/2})^2]\\ =FL[\omega_{n/2}^{k+n/2}]-\omega_n^k*FR[\omega_{n/2}^{k+n/2}]\\ =FL[\omega_{n/2}^k]-\omega_n^k*FR[\omega_{n/2}^k] \]

发现上下两式只有后面一项的符号不同,所以如果我们求出了 \(FL[x]\) 和 \(FR[x]\) 在 \(\omega_{n/2}^{0..n/2-1}\) 上的值,我们就可以求出 \(F[x]\) 在 \(\omega_n^{0...n-1}\) 上的值了,这样做一次的时间复杂度是 \(O(n)\) 的。同时我们可以发现,对于 \(FL[x]\) 和 \(FR[x]\) ,其实可以看做一个相同的子任务,再次递归下去求解。一共需要递归 \(log\ n\) 次。

但是此时我们又有一个问题:如果某个 \(n\) 不能被整除了怎么办?我们可以给它在后面补位,系数设为 0 就行。再观察一下我们递归到最下面一层时,系数的编号与第一层相比有什么特点。可以发现它是二进制位的反转,所以我们可以用数位DP的方法求出反转后的编号,这样就不用一遍一遍递归下去求了。

tr[i]=(tr[i>>1]>>1)|(i&1)?limit>>1:0

求多项式

通过上面的步骤,我们已经将 \(n\) 个点的值求出来了,接下来就要通过这 \(n\) 个点的值求多项式。

设点值构成的序列为 \(G[x]\) 。然后我们可以列出以下式子:

\[G[x]=\sum_{i=0}^{n-1}\omega_n^i*F[i] \]

然后可以得到以下结论:

\[F[x]*n=\sum_{i=0}^{n-1}\omega_n^{-i}*G[i] \]

这个式子的证明就是将上面的式子代入。此时我们发现,这个式子和我们求点值的式子十分相似,只是在单位根上有点区别。我们可以发现,在单位圆上, \(\omega_n^{-1}\) 所在的点和 \(\omega_n^1\) 所在的点,它们关于实轴对称。于是我们也可以用同样的方法来求上面的式子。最后记得除以 \(n\) 就是我们最终的多项式。

Code

#include<cstdio>
#include<cmath>
using namespace std;
const int N=3e6+5;const double PI=6.28318530717958646;
struct Date{
	double x,y;
	
	Date operator*(Date v) {
		Date res;res.x=x*v.x-y*v.y;res.y=x*v.y+y*v.x;return res;
	};
	Date operator+(Date v) {
		Date res;res.x=x+v.x;res.y=y+v.y;return res;
	};
	Date operator-(Date v) {
		Date res;res.x=x-v.x;res.y=y-v.y;return res;
	};
};Date f[N],g[N],w[N];
int tr[N],n,m;
void swap(double &x,double &y)
{
	double t=x;x=y;y=t;
}
void FFT(int flag)
{
	int i,j,len;Date t;
	w[0].x=1;w[0].y=0;
	w[1].x=cos(PI/n);w[1].y=sin(PI/n)*flag;
	for(i=1;i<n;i++)
	w[i]=w[i-1]*w[1];
	for(i=0;i<n;i++)
	{
		if(i<tr[i])
		{
			swap(f[i].x,f[tr[i]].x);swap(f[i].y,f[tr[i]].y);
		}
	}
	for(len=1;len<n;len=len<<1)
	{
		for(i=0;i<n;i+=2*len)
		{
			for(j=0;j<len;j++)
			{
				t=f[i+j+len]*w[n/len/2*j];
				f[i+j+len]=f[i+j]-t;f[i+j]=f[i+j]+t;
			}
		}
	}
}
int main()
{
	int i,h,len;
	scanf("%d%d",&n,&m);
	for(i=0;i<=n;i++)
	scanf("%lf",&f[i].x);
	for(i=0;i<=m;i++)
	scanf("%lf",&g[i].x);
	h=0;len=n+m;
	while((1<<h)<n+m+1)
	h++;
	n=1<<h;
	for(i=0;i<n;i++)
	tr[i]=(tr[i>>1]>>1)|((i&1)?n>>1:0);
	FFT(1);
	for(i=0;i<n;i++)
	{
		swap(f[i].x,g[i].x);swap(f[i].y,g[i].y);
	}
	FFT(1);
	for(i=0;i<n;i++)
	f[i]=f[i]*g[i];
	FFT(-1);
	for(i=0;i<=len;i++)
	printf("%d ",int(f[i].x/n+0.49));
	return 0;
}

标签:FR,变换,res,笔记,傅里叶,FL,多项式,Date,omega
From: https://www.cnblogs.com/Cyanwind/p/18368245

相关文章

  • 简单的linux系统学习笔记——08
    一、文件属性-权限属性33555523-rw-r--r--.1rootroot1586月72013/etc/hosts-rw-r--r-- 【rw-】:权限是属主的权限 【r--】:权限是属组的权限 【r--】:权限是其他人的权限r//读取权限 针对文件:含义是,可以打开文件,查看文件的内容 针对目录:含义是,可以查......
  • 简单的linux系统学习笔记——09
    一、用户分类1.root//皇帝用户,定制规则用户,系统高级管理员【uid,gid0】2.普通用户//有特定的权限,权限是root授予的【uid,gid大于1000】3.傀儡用户//没有家目录,不能登录系统;【0-999】二、用户相关的配置文件1.用户列表文件[root@c7-100~]#cat/etc/passwdroot:x......
  • C++图笔记(三)有向无环图(及最小生成树(略))以及剩下的排序
    目录一,定义:1,有向无环图 2,拓朴排序 1,每个顶点出现且仅仅出现一次。 2,若存在一条从顶点A到顶点B的路径,那么在序列中顶点A出现在顶点B的前面。二,DAG的性质性质1.  从任意一个起点进行dfs,必不会陷入死循环。性质2.  入度为0的点信息确定,删掉入度为0的点......
  • 【OpenCV_python】凸包检测 轮廓特征 直方图均衡化 模板匹配 霍夫变换
    凸包特征检测凸包就是图像的最小外接多边形,通过图像的轮廓点,找到距离最远的两个点的直线,根据直线找到距离最远的下一个点,直到所有的点被包围在多边形内读取图像二值化找图像的轮廓获取凸包点的坐标绘制凸包点convexHull获得图像的凸包点cv2.convexHull(points,hu......
  • 第一次项目搭建笔记&路由导航守卫&web前后端会话跟踪
    1.重新搭建后端项目在IDEA中重新创建一个JavaEE项目,记得勾选Webprofile之后在java文件中重新搭建分级的文件夹按照标准创建com.xxxx.dorm文件夹并创建dao(数据处理),filter(过滤器),model(模型),util(工具),web(服务端)等文件夹进行不同功能部分的分类搭建完基本的框......
  • 【杂乱笔记】Kmp字符串匹配算法
    KMP算法逻辑构建next数组:初始化next数组,用于存储每个位置的最长相同前后缀长度。遍历模式字符串patt如果当前字符与前缀字符匹配,增加前缀长度,并更新next数组。如果不匹配,使用next[prefix\_len-1]回退到上一个可能的前缀长度,继续比较。字符串匹配:初始......
  • HTML5+CSS3学习笔记补充——移动端网页+Bootstrap框架
    移动端网页和Bootstrap框架1.视口:用来约束HTML尺寸<!--视口标签是HTML骨架默认生成的设置网页宽度与逻辑分辨率(即设备)宽度一致--><metaname="viewport"content="width=device-width,initial-scale=1.0">2.二倍图:防止设计稿图片在高分辨率屏幕下模糊失真3.......
  • Java基础语法笔记
    Java基础语法1.注释,标识符,关键字注释linecomment://blockcomment:/**/javadoc:/***/标识符、关键字数据类型 ......
  • 《软件性能测试分析与调优实践之路》(第2版) 读书笔记(二)总体介绍(下)-真正从性能分析与
    《软件性能测试分析与调优实践之路》(第2版) 是清华大学出版社出版的一本图书,作者为张永清,全书共分为9章,如下图所示 图书介绍:《软件性能测试分析与调优实践之路》(第2版) 本文是接着 《软件性能测试分析与调优实践之路》(第2版)读书笔记(一)总体介绍(上)-真正从性能分析与调......
  • SpringMVC笔记
    目录一、SpringMVC简介1、什么是MVC2、什么是SpringMVC3、SpringMVC的特点二、HelloWorld1、开发环境2、创建maven工程a>添加web模块b>打包方式:warc>引入依赖3、配置web.xmla>默认配置方式b>扩展配置方式4、创建请求控制器5、创建springMVC的配置文件6、测试HelloWorlda>实现对首......