码迷,mamicode.com
首页 > 编程语言 > 详细

几种快速傅里叶变换(FFT)的C++实现

时间:2015-09-14 07:01:26      阅读:243      评论:0      收藏:0      [点我收藏+]

标签:

 

几种快速傅里叶变换(FFT)的C++实现

 

        DFT的的正变换和反变换分别为(1)和(2)式。假设有N个数据,则计算一个频率点需要N次复数乘法和N-1次复数加法,整个DFT需要N*N次复数乘 法和N(N-1)次复数加法;由于一次的复数乘法需要进行4次的实数乘法和2次的复数加法,一次的复数加法需要两次的实数加法,因此整个DFT需要 4*N*N次的实数乘法和2*N(N-1)+2*N*N≈4*N*N次的复数加法。当N比较大时,所需的计算工作量相当大,例如N=1024时大约需要 400万次乘法运算,对于实时信号处理来说,将对计算设备提出非常苛刻的要求,于是就提出如何能够减少计算DFT的运算量的问题。
1965年,库力和图基在《计算数学》杂志上发表《机器计算傅立叶级数的一种算法》,此文是最早提出FFT算法的。此后关于DFT的快速算法称为人们研究的热点课题,也正是FFT算法的出现,才使得数字信号处理能够应用于实时场合并进一步推动数字信号处理的发展和应用。

 技术分享

大多数的FFT算法都是利用(3)式的周期性、共轭对称性、可压缩和扩展性的特点来压缩计算量。

1)、根据DFT定义进行计算的代码

技术分享//Data为输入数据指针,Log2N=log2(length),flag=-1时为正变换,flag=1时为反变换,变换结果同样由指针Data指向的空间返回
void dft(complex<double>*Data,int Log2N,int flag)
技术分享技术分享...{
技术分享        int i,j,length;
技术分享        complex<double> wn;
技术分享        length=1<<Log2N;
技术分享        complex<double>*temp=new complex<double>(length);
技术分享 技术分享   for(i=0;i<length;i++)
技术分享   ...{
技术分享               temp[i]=0;
技术分享               for(j=0;j<length;j++)
技术分享技术分享       ...{
技术分享                    wn=complex<double>(cos(2.0*pi/length*i*j),sin(flag*2.0*pi/length*i*j));
技术分享                    temp[i]+=Data[j]*wn;    
技术分享                }           
技术分享       }
技术分享       if(flag==1)
技术分享           for(i=0;i<length;i++)
技术分享               Data[i]=temp[i]/length;
技术分享        delete[] temp;
技术分享}  

2)、基2时间抽选FFT

把时域的数字信号序列按照奇偶进行分组计算,可以进行如下的变换,从变换结果可以知道,一个长度为NDFT可以变换成长度为N/2的两个子序列的组合。依次类推,可以直到转为N/22点的傅立叶变化的组合。不过这时的输入应该为以2为基的倒位序。

技术分享

由于经过多次的奇偶抽选,输入数据要按照基2倒序的原则进行重排,输出数据为正常顺序,倒序算法另外叙述。下面首先用递归的形式进行算法的描述,由于递归方法没有充分利用DIT2算法的优点---原位计算,因此递归形式只是为了描述的清晰。

 

技术分享void dit2rec(complex<double>*InData,complex<double>*OutData,int length,int sign)
技术分享技术分享...{
技术分享   complex<double>*EvenData=new complex<double>(length/2);
技术分享   complex<double>*OddData  =new complex<double>(length/2);
技术分享   complex<double>*EvenResult=new complex<double>(length/2);
技术分享   complex<double>*OddResult=new complex<double>(length/2);
技术分享   int i,j;
技术分享   if(length==1)
技术分享技术分享   ...{
技术分享      OutData[0]=InData[0]/N;
技术分享      return;
技术分享   }
技术分享  for(i=0;i<length/2;i++)
技术分享技术分享  ...{
技术分享    EvenData[i]=InData[2*i];
技术分享    OddData[i]=InData[2*i+1];
技术分享  }
技术分享  dit2rec(EvenData,EvenResult,length/2,sign);
技术分享  dit2rec(OddData,EvenResult,length/2,sign);
技术分享  for(i=0;i<length/2;i++)
技术分享技术分享  ...{
技术分享    OutData[i]=EvenData+OddData*complex<double>(cos(2*pi*i/length),sin(sign*2*pi*i/length));
技术分享    OutData[i+length/2]=EvenData- OddData*complex<double>(cos(2*pi*i/length),sin(sign*2*pi*i/length));   }   delete[] EvenData,OddData,EvenResult,OddResult;   return; }
 
非递归实现如下(现不考虑输入的倒序数问题):
技术分享void dit2(complex<double>*Data,int Log2N,int sign)
技术分享技术分享...{
技术分享   int i,j,k,step,length;
技术分享   complex<double> wn,temp,deltawn;
技术分享   length=1<<Log2N;
技术分享   for(i=1;i<=Log2N;i++)
技术分享技术分享   ...{
技术分享      wn=1;step=1<<i;deltawn=complex<double>(cos(2*pi/step),sin(sign*2.0*pi/step));
技术分享      for(j=0;j<step/2;j++)
技术分享技术分享      ...{        
技术分享        for(i=0;i<length/step;i++)
技术分享技术分享        ...{
技术分享           temp=Data[i*step+step/2+j]*wn;
技术分享           Data[i*step+step/2+j]=Data[i*step+j]-temp;
技术分享           Data[i*step+j]=Data[i*step+j]+temp;
技术分享         }
技术分享        wn=wn*deltawn;
技术分享      }
技术分享    }
技术分享    if(sign==-1)
技术分享       for(i=0;i<length;i++)
技术分享            Data[i]/=length;
技术分享 }
技术分享
 
i=1时,也就是第一次循环并没有必要进行复数运算,因为j只能取1wn为实数,这个时间可以节省。因此可以改进为:
技术分享void dit2(complex<double>*Data,int Log2N,int sign)
技术分享技术分享...{
技术分享   int i,j,k,step,length;
技术分享   complex<double> wn,temp,deltawn;
技术分享  length=1<<Log2N;
技术分享   for(i=0;i<length;i+=2)
技术分享技术分享   ...{
技术分享      temp=Data[i];
技术分享      Data[i]=Data[i]+Data[i+1];
技术分享      Data[i+1]=temp-Data[i+1];
技术分享   }
技术分享   for(i=2;i<=Log2N;i++)
技术分享技术分享   ...{
技术分享      wn=1;step=1<<i;deltawn=complex<double>(cos(2.0*pi/step),sin(sign*2.0*pi/step));;
技术分享      for(j=0;j<step/2;j++)
技术分享技术分享      ...{        
技术分享        for(i=0;i<length/step;i++)
技术分享技术分享        ...{
技术分享           temp=Data[i*step+step/2+j]*wn;
技术分享           Data[i*step+step/2+j]=Data[i*step+j]-temp;
技术分享           Data[i*step+j]=Data[i*step+j]+temp;
技术分享         }
技术分享         wn=wn*deltawn;
技术分享      }
技术分享   }
技术分享   if(sign==1)
技术分享   for(i=0;i<length;i++)
技术分享     Data[i]/=length;
技术分享}
技术分享
技术分享
3)、基2频率抽选FFT
技术分享
技术分享//DIF2的递归版本实现:
技术分享void dif2rec(complex<double>*InData,complex<double>*OutData,int length,int sign)
技术分享技术分享...{
技术分享   complex<double>* LData=new complex<double>(length/2);
技术分享   complex<double>* LResult=new complex<double>(length/2);
技术分享   complex<double>* RData=new complex<double>(length/2);
技术分享   complex<double>* RResult=new complex<double>(length/2);
技术分享   complex<double> temp;
技术分享   int i;
技术分享if(length==1)
技术分享技术分享   ...{
技术分享       OutData[0]=InData[0];
技术分享       return;
技术分享}
技术分享for(i=0;i<length/2;i++)
技术分享技术分享...{
技术分享   LData[i]=InData[i];
技术分享   RData[i]=InData[i+length/2];
技术分享}
技术分享for(i=0;i<length/2;i++)
技术分享技术分享...{
技术分享   temp=LData[i];
技术分享   LData[i]=LData[i]+RData[i];
技术分享   RData[i]=(temp-RData[i])* complex<double>(cos(2*pi*i/length),sin(sign*2*pi*i/length))
技术分享}
技术分享dit2rec(LData,LResult,length/2,sign);
技术分享dit2rec(RData,RResult,length/2,sign);
技术分享   for(i=0;i<length/2;i++)
技术分享技术分享   ...{       OutData[2*i]=LResult[i];       OutData[2*i+1]=RResult[i]; } return; }
技术分享//非递归实现如下(现不考虑输入的倒序数问题):
技术分享void dif2(complex<double>*InData,int r,int sign)
技术分享技术分享...{
技术分享int length=1<<r;
技术分享int i,j,k,step;
技术分享complex<double> temp,wn;
技术分享for(i=1;i<=r;i++)
技术分享技术分享...{
技术分享   step=1<<(r-i+1);
技术分享   wn=1;
技术分享   for(j=0;j<step/2;j++)
技术分享技术分享   ...{
技术分享      for(k=0;k<step/2;k++)
技术分享技术分享      ...{
技术分享         temp=InData[k*step+j];
技术分享         InData[k*step+j]=InData[k*step+j]-InData[k*step+step/2+j];
技术分享         InData[k*step+step/2+j]=(temp-InData[k*step+step/2+j])*wn;
技术分享}
技术分享wn=wn*complex<double>(cos(2*pi/step*j),sin(sign*2*pi/step*j));
技术分享}
技术分享}
技术分享}
技术分享

DIT一样,最外层的最后一个循环可以另外独立出来,因为最后一个循环没有必要进行复数运算,这样可以减少复数运算的次数。

基四时间抽选快速傅立叶算法

技术分享

版权声明:本文为博主原创文章,未经博主允许不得转载。

几种快速傅里叶变换(FFT)的C++实现

标签:

原文地址:http://www.cnblogs.com/timdes/p/4806019.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!