本文开始前我们先简单的介绍一下什么是THD?
2 s; `& y% E+ }- R! G& L/ Q" m [8 c' Z; m& B7 h: j
总谐波失真(Total Harmonic Distortion,简称THD)是衡量信号失真程度的重要指标,特别是在音频、电力电子设备和通信系统等领域。THD表示一个信号中谐波分量(除了基波之外的频率分量)相对于基波的比例。具体来说,THD是所有谐波分量的平方和与基波分量平方的比值的平方根,通常用百分比表示。
/ f# n4 r3 Q) V$ ^( {* {4 Z0 f4 C: Q" V+ B! {1 Y+ b
公式表示为:% Z0 `4 Z p! F* z3 _
, F/ Q- _$ P! _5 J1 H' Z/ i2 G2 O+ [/ x* d7 H+ r' s
其中的V1是基波幅度,V2,V3……Vn是谐波幅度。
, i3 e3 Y4 J" W: F. m( ^7 E4 A6 k# w& r" q' Y: m0 U! u
这个公式很好理解,一个信号的谐波幅度平方和占谐波频率的百分比。
5 q, ~/ x+ i, I+ [9 K
: c# l3 h. p& s: i- y2 R那么如何获取基波幅度和谐波幅度,就是我们的FFT需要进行的事情。
6 F( C) R2 r0 d0 ~% E% Z5 ~ N0 Q- G
利用FFT我们可以获取信号的频率谱。$ Y7 h. C K; K( I: L
( w: \1 \0 f: ~0 M0 |
0 [/ ~; C3 S4 s( i$ }. y6 f0 d* m2 x
$ M. e c4 B, [) u例如一段信号方波信号经过FFT之后,这些突出的方波信号就是谐波的幅值,而第一段(不包括第一个点)的则是其基波幅度。但是这里不包含第一个点,因为那个点是直流的幅度(两倍)。" |/ K" w- c& h$ c% f
8 q) x: S7 L. e' u% ^
因此我们只需要进行FFT之后统计谐波的平方和即可获得我们的THD。6 J; I: V6 q' J9 Q4 V1 ~; W9 {9 y
1 h) q" B O X& f% j7 O7 ]) r
快速傅里叶变换5 n$ G$ U2 }1 X" d5 q
关于FFT我们在之前的文章中有过介绍,我们在Keil中可以利用DSP库来进行快速的FFT(手搓的FFT算法效率通常很低)。我们在之前的文章中介绍过如何使用ADC+DMA的快速采样配合DSP库进行FFT。
! S$ I; Q3 a" O. w6 W+ `, P. D1 J3 b+ \4 U+ J3 k1 @
$ N) B$ u( V4 ~) E7 A2 M: Q
$ Q8 I. k# Z$ q! t6 I
STM32的DMA采样+FFT时域分析(STM32F407); X7 O/ M- h0 u+ @
z; L9 u8 g3 d) t! W' b: ~我们同样的在这篇文章的基础上进行。) G' a* Q4 N) V/ W, _9 B
- #define ADCLenth 1024
3 C; r+ o1 Z' A* H. j/ | - uint16_t ADCValue[ADCLenth];! a, G6 q" X% @: N6 h; A( Q# F% o( ]
- arm_cfft_radix4_instance_f32 scfft;//定义scfft结构体( I8 C' F+ V+ o F$ S
- float FFT_InputBuf[ADCLenth*2]; //FFT输入数组3 C) w- X7 \, o" G/ O' ~
- float FFT_OutputBuf[ADCLenth]; //FFT输出数组
复制代码
/ V! b, D! u$ Z, v, M! [, S4 d我们定义一个FFT的结构体变量,以及存放ADC采样结果的数组还有FFT的输入输出数组,这里的输入数组之所以长度翻倍是因为对于FFT的输入来说是一个复数即包含实部和虚部,因此长度是两倍。 }7 G- n+ i9 [5 ?/ \
- for(int i=0; i < ADCLenth; i++)- j( T) t- l7 G# C
- {/ h L/ O- A8 a9 t
- FFT_InputBuf[2*i]=(ADCValue[i] )*3.3/4096; //实部
7 G* [ O4 ]: z; p( X' u. ] - FFT_InputBuf[2*i+1]=0; //虚部
; ~1 ^9 F7 x# X r9 k - }8 x4 ?: O5 _& X2 _0 ^: b
- arm_cfft_radix4_f32(&scfft,FFT_InputBuf); / a0 Y5 D0 H6 L3 ]
- arm_cmplx_mag_f32(FFT_InputBuf,FFT_OutputBuf,ADCLenth); //取模得幅值
. e% f' j8 q* N4 i8 A. N/ K/ i9 s, h G -
. c( u2 }! S. j8 b% S) p - 9 I% u( ^- g$ ^2 M8 v* M
- for(int i = 2;i<ADCLenth/2;i++)
, K% }) p0 ] _5 a8 O - {
; p/ T/ @' g/ H: y7 G - if(FFT_OutputBuf[i]>maxValue)% x, ^6 x5 M5 c/ `
- {
: ?# N7 W# _( K- r# H -
$ ], `" d# o6 _/ Q9 N, ? - maxValue = FFT_OutputBuf[i];
- d. `! a' q& Y2 O: K - max = i;, ?& k! r& c- J. Y' z5 ]1 s! t
-
3 J* o' Z8 R4 ]: F - }
; I, g4 d2 T) J/ {$ ]$ d! a! `9 J- e - }
复制代码
! a9 A9 ]5 X" K' L. c f一轮ADC采样结束之后,我们将其的实部信号和虚部信号(0)存放FFT的输入数组,之后执行快速傅里叶变换获得FFT的模值,模值存放在FFT_OutputBuf中。
' h2 c0 c, i' j$ l: l# P+ |& v5 ?( g K: x m; }; M- k
我们通过一个比较循环来寻找FFT结果的最大值,这里我们忽略前几个元素尤其是索引为0的位置,因为他是直流分量的模值。
0 L, z! h5 F) C3 n
, b- T( E- @7 d7 J+ }/ k
, n) Q: o; r/ }# @ G6 y
% z z! x0 s2 z- k3 c! @之后我们就需要统计各个谐波的幅度。
) v, i" Q( {, M0 f- o- float FindMax(float * fft,int index,int wind)) _$ s7 l4 h4 M. C
- {1 t& e4 G, w) N3 x
- int max = 0;$ [2 D; p: Y2 ^/ a
- for(int i = index-wind ;i <index+wind;i++); ]: I" }9 O7 Z4 @/ H
- {0 I$ Q/ V- g; N! O- r: \
- if(fft[i]>max)
) W$ ~0 _7 o* Q$ c3 r! z+ t) k - {
2 h$ R/ t, P l2 @1 Q% ? - max =fft[i];, A( g i/ L7 S& d4 [& _3 B
- }
P# d0 {% G* H -
- t) P! ~& W" l+ A# O1 N) ` - }6 G/ ?! ^' m7 {5 m# f
- return max;
1 w. F; Y! y4 \# ` - % @$ D# K2 {9 E! ]
- }
复制代码
3 P" E% s# F6 U% H( p: U( `我们定义一个函数来寻找某点附近的最大值,这里之所以要寻找某点附近的最大值在这里说明一下。
0 S! O7 L* U+ @3 r5 b4 O: ~" p
) U# a+ A- [2 w+ D b Z# a/ gFFT的索引和频率有关系,每个索引对应的频率之差为:采样频率/采样长度。我们的采样长度是1024,而采样频率是20kHZ,因此索引差为对应的频率差为19.53HZ,因为根据计算,1000HZ的频率对应的索引为51.2而由于索引只能是整数,因此在频谱上基波的最大值并不会是51.2而是51,因为int类型会抛弃小数。8 s R. W$ x5 O6 v' A7 s" T
3 a$ }: ^6 a9 v3 l* S: t
所以我们实际统计的是984.3HZ或者1015.56HZ的频率,这样子我们计算谐波幅度进行翻倍的时候并不会是刚刚好好的对应的2KHZ的频率,而是其最大幅度会发生便宜。, e3 e; M5 U& w9 F
# I0 P/ M" @; T
因此我们因为是寻找我们认为的谐波索引的附近最大值。
/ m: w- s( C' }8 b- for(int i = max;i<ADCLenth/2;i+=max)
- j \( V0 J( a' q; _$ b' \7 b; ?2 ? - {
! G4 r* F: H2 K6 \' j' L' Q( y* r - maxW = FindMax(FFT_OutputBuf,i,max*0.3);& \) ]1 v5 _3 Y, f) @3 @3 c
- all =all + maxW*maxW;# a+ ?" d4 ?) h' V6 o
- }# Z& Q: b/ k' _% y
- all = all - maxValue*maxValue;
7 U4 P) W& S4 @/ Q -
$ ?* e- Q. y* f/ J+ ] - DHT = sqrtf(all)/maxValue;/ t( L4 j6 k! n& e. |" c4 X |
- printf("DHT:%f%c\r\n",DHT*100,'%');: m& h3 L: d2 E! l! W7 M/ w& B. M
- all = 0;
2 e0 Y0 v# V0 M5 l - max = 2;
* C# h" e/ U% C% t8 d" L - maxValue = 0;
复制代码 ) ^( t0 u3 ]. O4 R6 ?- s
这样子就是统计我们的THD的值,之后将其打印,这里计算谐波幅度的时候我把基波的幅度也加了上去,我们将其去除。" `) t7 a/ |6 {. s* f* S2 @6 V1 d
: I+ K3 J! t9 R! N8 E% Q) L
% {2 r; B9 ]' R9 p% @# v! V' {8 Z- P- v/ ^5 K5 W0 Z
可以看到外面的THD(图中打错了)计算外面的方波频率的THD在40%左右。
7 P. @# C. q9 M) B2 I0 g# j我们利用MatLab生成一下方波信号之后用FFT来看一下理论结果。
" H1 U2 t8 ]) O: S9 U7 {
; a6 k4 ^2 ^" g f# L6 i' Z% N
4 m7 o: B5 C! g7 w0 u4 a! q
' |, M) r0 U+ s. C1 y
. q* H9 H; \) l' q F
! _+ o/ ?) T; D( t! O f0 m9 `其统计到5次谐波对应的值计算出的THD的值43%,这和我们的计算结果也很接近了。8 K$ S) d0 N& H' V+ {
我们同样的看一下手机上的显示内容。
& Q5 ?5 a0 J( B4 d4 ~4 O2 G5 ~) H+ }$ I6 M. O
# n- Z: T- ], H5 h/ W
- r: i* }) n. k, o6 a! l( d% J总谐波失真为42.7%4 F l a* T: E* r
$ S, ^) z. C( x( l' ]! S3 ^5 {7 o" p x" ]
}6 Q4 a* H( [! D; o
转载自:电路小白0 O- l/ Y) r, ^6 a1 D: t. E, E1 W
如有侵权请联系删除
6 k( ]! A0 n4 C9 J
" q$ V4 r% s2 U6 _) J+ u" U. w/ [' x3 J1 d
; Z$ p% e4 D0 j
; g6 s* l/ J( g( |: y* d8 d
|