快速傅里葉變換FFT的C程序代碼實(shí)現(xiàn)
2016年10月08日 16:38 來源:互聯(lián)網(wǎng) 作者:辰光 我要評論(0)
快速傅里葉變換(Fast Fourier Transform)是離散傅里葉變換的一種快速算法,簡稱FFT,通過FFT可以將一個信號從時域變換到頻域。
模擬信號經(jīng)過A/D轉(zhuǎn)換變?yōu)?a target="_blank">數(shù)字信號的過程稱為采樣。為保證采樣后信號的頻譜形狀不失真,采樣頻率必須大于信號中最高頻率成分的2倍,這稱之為采樣定理。
假設(shè)采樣頻率為fs,采樣點(diǎn)數(shù)為N,那么FFT結(jié)果就是一個N點(diǎn)的復(fù)數(shù),每一個點(diǎn)就對應(yīng)著一個頻率點(diǎn),某一點(diǎn)n(n從1開始)表示的頻率為:fn=(n-1)*fs/N。
舉例說明:用1kHz的采樣頻率采樣128點(diǎn),則FFT結(jié)果的128個數(shù)據(jù)即對應(yīng)的頻率點(diǎn)分別是0,1k/128,2k/128,3k/128,…,127k/128 Hz。
這個頻率點(diǎn)的幅值為:該點(diǎn)復(fù)數(shù)的模值除以N/2(n=1時是直流分量,其幅值是該點(diǎn)的模值除以N)。
二、傅里葉變換的C語言編程
1、對于快速傅里葉變換FFT,第一個要解決的問題就是碼位倒序。
假設(shè)一個N點(diǎn)的輸入序列,那么它的序號二進(jìn)制數(shù)位數(shù)就是t=log2N.
碼位倒序要解決兩個問題:①將t位二進(jìn)制數(shù)倒序;②將倒序后的兩個存儲單元進(jìn)行交換。
如果輸入序列的自然順序號i用二進(jìn)制數(shù)表示,例如若最大序號為15,即用4位就可表示n3n2n1n0,則其倒序后j對應(yīng)的二進(jìn)制數(shù)就是n0n1n2n3,那么怎樣才能實(shí)現(xiàn)倒序呢?利用C語言的移位功能!
程序如下,我不多說,看不懂者智商一定在180以下!
復(fù)數(shù)類型定義及其運(yùn)算
#define N 64 //64點(diǎn)
#define log2N 6 //log2N=6
/*復(fù)數(shù)類型*/
typedef struct
{
float real;
float img;
}complex;
complex xdata x[N]; //輸入序列
/*復(fù)數(shù)加法*/
complex add(complex a,complex b)
{
complex c;
c.real=a.real+b.real;
c.img=a.img+b.img;
return c;
}
/*復(fù)數(shù)減法*/
complex sub(complex a,complex b)
{
complex c;
c.real=a.real-b.real;
c.img=a.img-b.img;
return c;
}
/*復(fù)數(shù)乘法*/
complex mul(complex a,complex b)
{
complex c;
c.real=a.real*b.real - a.img*b.img;
c.img=a.real*b.img + a.img*b.real;
return c;
}
/***碼位倒序函數(shù)***/
void Reverse(void)
{
unsigned int i,j,k;
unsigned int t;
complex temp;//臨時交換變量
for(i=0;i
k=i;//當(dāng)前第i個序號
j=0;//存儲倒序后的序號,先初始化為0
for(t=0;t
j<<=1;
j|=(k&1);//j左移一位然后加上k的最低位
k>>=1;//k右移一位,次低位變?yōu)樽畹臀?br /> }
if(j>i)//如果倒序后大于原序數(shù),就將兩個存儲單元進(jìn)行交換(判斷j>i是為了防止重復(fù)交換)
{
temp=x;
x=x[j];
x[j]=temp;
}
}
}
2、第二個要解決的問題就是蝶形運(yùn)算

?
第m級蝶形運(yùn)算,每個蝶形的兩節(jié)點(diǎn)“距離”L=2m-1。
?、趯τ?6點(diǎn)的FFT,第1級有16組蝶形,每組有1個蝶形;第2級有4組蝶形,每組有2個蝶形;第3級有2組蝶形,每組有4個蝶形;第4級有1組蝶形,每組有8個蝶形。由此可推出,
對于N點(diǎn)的FFT,第m級有N/2L組蝶形,每組有L=2m-1個蝶形。
③旋轉(zhuǎn)因子

以16點(diǎn)FFT為例,第m級第k個旋轉(zhuǎn)因子為



為提高FFT的運(yùn)算速度,我們可以事先建立一個旋轉(zhuǎn)因子數(shù)組,然后通過查表法來實(shí)現(xiàn)。
complex code WN[N]=//旋轉(zhuǎn)因子數(shù)組
{ //為節(jié)省CPU計算時間,旋轉(zhuǎn)因子采用查表處理
//★根據(jù)實(shí)際FFT的點(diǎn)數(shù)N,該表數(shù)據(jù)需自行修改
//以下結(jié)果通過Excel自動生成
// WN[k].real=cos(2*PI/N*k);
// WN[k].img=-sin(2*PI/N*k);
{1.00000,0.00000},{0.99518,-0.09802},{0.98079,-0.19509},{0.95694,-0.29028},
{0.92388,-0.38268},{0.88192,-0.47140},{0.83147,-0.55557},{0.77301,-0.63439},
{0.70711,-0.70711},{0.63439,-0.77301},{0.55557,-0.83147},{0.47140,-0.88192},
{0.38268,-0.92388},{0.29028,-0.95694},{0.19509,-0.98079},{0.09802,-0.99518},
{0.00000,-1.00000},{-0.09802,-0.99518},{-0.19509,-0.98079},{-0.29028,-0.95694},
{-0.38268,-0.92388},{-0.47140,-0.88192},{-0.55557,-0.83147},{-0.63439,-0.77301},
{-0.70711,-0.70711},{-0.77301,-0.63439},{-0.83147,-0.55557},{-0.88192,-0.47140},
{-0.92388,-0.38268},{-0.95694,-0.29028},{-0.98079,-0.19509},{-0.99518,-0.09802},
{-1.00000,0.00000},{-0.99518,0.09802},{-0.98079,0.19509},{-0.95694,0.29028},
{-0.92388,0.38268},{-0.88192,0.47140},{-0.83147,0.55557},{-0.77301,0.63439},
{-0.70711,0.70711},{-0.63439,0.77301},{-0.55557,0.83147},{-0.47140,0.88192},
{-0.38268,0.92388},{-0.29028,0.95694},{-0.19509,0.98079},{-0.09802,0.99518},
{0.00000,1.00000},{0.09802,0.99518},{0.19509,0.98079},{0.29028,0.95694},
{0.38268,0.92388},{0.47140,0.88192},{0.55557,0.83147},{0.63439,0.77301},
{0.70711,0.70711},{0.77301,0.63439},{0.83147,0.55557},{0.88192,0.47140},
{0.92388,0.38268},{0.95694,0.29028},{0.98079,0.19509},{0.99518,0.09802}
};
3、算法實(shí)現(xiàn)
我們已經(jīng)知道,N點(diǎn)FFT從左到右共有l(wèi)og2N級蝶形,每級有N/2L組,每組有L個。所以FFT的C語言編程只需用3層循環(huán)即可實(shí)現(xiàn):最外層循環(huán)完成每一級的蝶形運(yùn)算(整個FFT共log2N級),中間層循環(huán)完成每一組的蝶形運(yùn)算(每一級有N/2L組),最內(nèi)層循環(huán)完成單獨(dú)1個蝶形運(yùn)算(每一組有L個)。
/***【快速傅里葉變換】***/
void FFT(void)
{
unsigned int i,j,k,l;
complex top,bottom,xW;
Reverse(); //碼位倒序
for(i=0;i
l=1< for(j=0;j
for(k=0;k
xW=mul(x[j+k+l],WN[N/(2*l)*k]); //碟間距為l
top=add(x[j+k],xW); //每組的第k個蝶形
bottom=sub(x[j+k],xW);
x[j+k]=top;
x[j+k+l]=bottom;
}
}
}
}
三、FFT計算結(jié)果驗(yàn)證
隨便輸入一個64點(diǎn)序列,例如
x[N]={{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0},{8,0},{4,0},{1,0},{3,0},{2,0},{5,0}};
在Keil中Debug查看數(shù)組變量x的FFT計算結(jié)果并與MATLAB計算結(jié)果進(jìn)行比對,可以發(fā)現(xiàn)非常準(zhǔn)確,說明程序編寫正確!
上周熱點(diǎn)文章排行榜
上周資料下載排行榜
論壇熱帖
- LF開頭請問這個是什么封裝元器件 jf_77000477
- 【飛凌嵌入式OK3588J-C開發(fā)板體驗(yàn)】OK3588J-C開發(fā)板開箱評測 jf_43382582
- 【書籍評測活動NO.52】基于大模型的RAG應(yīng)用開發(fā)與優(yōu)化 ElecFans小喇叭
- 求一份CS32L010的相關(guān)資料(數(shù)據(jù)手冊、用戶手冊、Pack包和例程等),謝謝 jf_43621189
- 請問如何關(guān)閉獨(dú)立看門狗 jf_39582415
- 【「從算法到電路—數(shù)字芯片算法的電路實(shí)現(xiàn)」閱讀體驗(yàn)】+閱讀濾波器章節(jié)部分筆記 james_87
熱門博文
論壇熱帖
FFT技術(shù)應(yīng)用
FFT資料下載
- 使用ADC FFT數(shù)據(jù)進(jìn)行輸入阻抗測量
- TMS320VC5505、TMS320C5505和TMS320C5515 DSP上的FFT實(shí)現(xiàn)
- 使用突發(fā)序列器模式在ADS8686S上進(jìn)行平均來降低相位延遲
- 使用DSPLIB FFT實(shí)現(xiàn)實(shí)現(xiàn)實(shí)際輸入,無需數(shù)據(jù)縮放
- 一種高動態(tài)與低信噪比條件下的載波同步方法
- 基于單片機(jī)的FFT算法分析與實(shí)現(xiàn)
- HPM6000系列微控制器DSP/FFT使用介紹
- 在AI引擎上實(shí)現(xiàn)逐塊可配置的快速傅里葉變換應(yīng)用說明
- 數(shù)字信號處理的六個實(shí)驗(yàn)案例
- Raspberry Pi Pico上的ADC采樣和FFT
C程序技術(shù)應(yīng)用
C程序資料下載
- 最簡單的C程序設(shè)計
- C語言全部章節(jié)復(fù)習(xí)題與解答
- 200個經(jīng)典C程序【源碼】
- C語言概述及如何上機(jī)運(yùn)行C程序
- C語言學(xué)習(xí)課件
- 基于單片機(jī)便攜式太陽能充電器系統(tǒng)設(shè)計
- 473【畢設(shè)課設(shè)】基于STM32單片機(jī)的智能家居煙霧co檢測系統(tǒng)設(shè)計
- C程序設(shè)計(非計算機(jī)專業(yè)類) 實(shí)驗(yàn)指導(dǎo)書
- 51單片機(jī)的直流電機(jī)PWM調(diào)速控制系統(tǒng)(附Proteus仿真+C程序等全套資料)
- 單片機(jī)學(xué)習(xí)教程之C程序的簡單介紹
熱評
- IR將功率半導(dǎo)體觸角伸往消費(fèi)市場
- 多功能算術(shù)/邏輯運(yùn)算單元(ALU) ,什么是多功能
- 動態(tài)ip、靜態(tài)ip、pppoe撥號的區(qū)別
- ARM與MIPS的比較
- ThunderBolt端口驅(qū)動及NET改WAP方法
- ds18b20中文資料詳解
- 滴滴人臉識別怎么破解
- 超級計算機(jī)榜單重新排名 中國天河二號已淪為世界第
- 電阻色環(huán)表_色環(huán)電阻識別方法
- iphone6概念機(jī)圖片曝光_iphone6上市時
博文
帖子
- 【書籍評測活動NO.51】具身智能機(jī)器人系統(tǒng) | 了解AI的下一個浪潮! ElecFans小喇叭
- 請教關(guān)于CS1239低側(cè)采樣如何獲得使用更高的Gain? jf_94221193
- 高頻條件下的耦合線圈出現(xiàn)負(fù)值的原因是什么 jf_42363055
- 請問如何關(guān)閉獨(dú)立看門狗 jf_39582415
- ads1291雙電源供電時,Thermal Pad接AVSS嗎? ggfx
- 【米爾-Xilinx XC7A100T FPGA開發(fā)板試用】+03.SFP光口測試(zmj) 卿小小_9e6
- DAC7621的reference性能精度,可以使用外部輸入ref嗎? 萬物死
- 【「HarmonyOS NEXT啟程:零基礎(chǔ)構(gòu)建純血鴻蒙應(yīng)用」閱讀體驗(yàn)】+1-7章有感 夜孤影
- DIY了一臺無人機(jī),用全志T113芯片 文小二
- 電子產(chǎn)品結(jié)構(gòu)與導(dǎo)熱材料解決方案 jf_86221244
用戶評論
查看全部 條評論
查看全部 條評論>>