第14章 富里哀分析 象富里哀级数,富里哀变换以及它们离散时间相应部分构成了信号处理的基础。为了便于这类问题的分析,MATLAB提供了函数fft,ifft,fft2,ifft2和fftshift。这类函数集执行一维和二维离散富里哀变换及其逆变换。这些函数允许人们完成很多信号处理任务。除此之外,还可在可选的信号处理工具箱中得到其他扩展的信号处理工具。 因为信号处理包含如此广泛的领域,甚至要说明用MATLAB中离散富里哀变换函数可解决的这类小问题,就超出了本书的范围。因此,这里将只介绍用函数fft近似连续时间信号的富里哀变换的一个例子。此外,还将讨论《精通MATLAB工具箱》中处理富里哀级数的函数集。 14.1 快速富里哀变换 在MATLAB中,函数fft计算一个信号的离散富里哀变换。在数据的长度是2的幂次或质因数的乘积的情况下,就用快速富里哀变换(FFT)来计算离散富里哀变换。当数据长度是2的幂次时,计算速度显著增加,因此,只要可能,选择数据长度为2的幂次或者用零来填补数据,使得数据长度等于2的幂次显得非常重要。在《MATLAB参考指南》中可找到有关该问题的讨论。 MATLAB中实现的快速富里哀变换,是按照工科教材中常使用的方法。 F(k)=FFT{f(n)}  因为MATLAB不允许零下标,所以移动了一个下标值。  相应的逆变换为:  为了说明FFT的使用,考虑估计连续信号的富里哀变换的问题。  解析上,该富里哀变换为:  虽然在这种情况下,由于知道了富里哀变换的解析结果,再运用FFT没有多大的实用价值,但这个例子说明了对不常见的信号,特别是那些解析上难以找到富里哀变换的信号,一个估计富里哀变换的方法。下面的MATLAB语句用FFT估计F(w),并且用图形把所得到结果与上面的解析表达式的结果进行比较: >>N=128; % choose a power of 2 for speed >>t=linspace(0, 3, N); % time points for function evaluation >>f=2*exp(-3*t); % evaluate the function and minimize aliasing:f(3)~0 >>Ts=t(2)-t(1); % the sampling period >>Ws=2*pi/Ts; % the sampling frequency in rad/sec >>F=fft(f); % compute the fft >>Fp=F(1 : N/2+1)*Ts;  图14.1 富里哀变换两种结果的比较 仅从F中取正频率分量,并且乘以采样间隔计算F(w)。 >>W=Ws*(0 : N/2)/N 它建立了连续频率轴,该轴起始于0,终止于奈魁斯特(Nyquist)频率Ws/2, >>Fa=2./(3+j*w); % evaluate analytical Fourier transform >>plot(W, abs(Fa), W, abs(Fp), ‘ + ‘ ) % generate plot, ‘ + ‘ mark fft results >>xlabel(‘ Frequency, Rad/s ‘),ylabel(‘ |F(w)| ‘) MATLAB提供了大量的完成一般信号处理任务的函数。它们列于表14.1: 表14.1 信号处理函数  conv 卷积  conv2 2维卷积  fft 快速富里哀变换  fft2 2维快速富里哀变换  ifft 逆快速富里哀变换  ifft2 2维逆快速富里哀变换  filter 离散时间滤波器  filter2 2维离散时间滤波器  abs 幅值  angle 四个象限的相角  unwrap 在360°边界清除相角突变  fftshift 把FFT结果平移到负频率上  nextpow2 2的下一个较高幂次   14.2 富里哀级数 MATLAB本身没有特别关于富里哀级数分析和处理的函数。不过,通过创建M文件函数,可容易加上这些函数。在这一节,将介绍《精通MATLAB工具箱》中富里哀级数函数。在介绍之前,首先定义实周期信号f(t)的富里哀级数表示形式。 给出富里哀级数的复指数形式为:  式中的富里哀级数的系数是:  且基频为。式中T0满足f(t+ T0)=f(t)。 给出富里哀级数的三角数形式为:  式中的富里哀级数的系数是:    且基频为。式中T0满足f(t+ T0)=f(t)。 表14.2 精通MATLAB的富里哀级数函数  fsderiv(Kn,Wo) 富里哀级数的微分  fseval(Kn,t,Wo) 计算富里哀级数  fsfind(‘ fname ‘,T,N) 寻找时间函数的富里哀级数的系数  [An,Bn,Ao]=fsform(Kn) Kn=fsform(An,Bn,Ao) 富里哀级数不同形式之间的转换  fsharm(Kn,i) 提取特殊的富里哀级数的谐波  fsmsv(Kn) 计算信号的均方根值  fsresize(Kn,N) 重新调整富里哀级数的系数向量的大小  fsresp(Num,Den,Un,Wo) 线性系统对输入富里哀级数Un的富里哀级数响应  fsround(Kn) 设置次要的富里哀级数的系数为0  fswindow(N, ‘ type ‘) fswindow(Kn, ‘ type ‘) 产生一个窗口的函数,使吉布(Gibb)现象极小   上述两种形式中,一般复指数的富里哀级数更容易进行解析上处理。而三角富里哀级数则提供了更多的直观理解,更容易将正弦和余弦波形可视化。由于这个原因,《精通MATLAB工具箱》中的富里哀级数函数一般假定为复指数形式。不过,该工具箱提供了这两种富里哀级数形式之间转换的函数。表14.2总结了《精通MATLAB工具箱》中的富里哀级数函数。 因为有无穷多个谐波或富里哀级数的系数,有必要对富里哀级数截尾,只考虑有限个谐波。如果考虑N个谐波,MATLAB中富里哀级数表示成一个长度为2N+1的行向量,向量中的元素为复指数的富里哀级数的系数。该向量包含以升序排列的富里哀级数系数,即: 为了说明这些函数的使用,考虑寻找锯齿信号的富里哀级数表示(见图14.2)。  图14.2 锯齿信号 虽然很容易找到这个信号的富里哀级数的系数,但可用函数fsfind近似求解这些系数。 function [Fn, nwo, f, t]=fsfind(fun, T, N, P) % FSFIND Find Fourier Series Approximation. % Fn=FSFIND(FUN, T, P) computes the Complex Exponential % Fourier Series of a signal described by the function ' FUN '. % FUN is the character string name of a user created M-file function. % the unction is called as f=FUN(t) where t is a vector over % the range 0<=t<=T. % % The FFT is used. Choose sufficient harmonics to minimize aliasing. % % T is the period of the function. N is the number of harmonics. % Fn is the vector of FS coefficients. % % [Fn, nWo, t, p]=FSFIND(FUN, T, P) returns the frequencies associated % with Fn in nWo and return values of the function FUN % in f evaluated at the points int t over the range 0<=t<=T. % % FSFIND(FUN, T, P) passes the data in P to function FUN as % f=FUN(t, p). This allows parameters to be passed to FUN. % Copyrigth (c) 1996 by Prentice-Hall,Inc. n=2*N; t=linspace(0, T, n+1); if nargin==3 f=feval(fun, t); else f=feval(fun, t, P); end Fn=fft(f(1 : n)); Fn=[0 conj(Fn(N : -1 : 2)) Fn(1 : N) 0]/n; nwo=2*pi/T*(-N : N); 根据上述描述,必须先建立一个函数以计算周期内锯齿信号值。这样,我们有: function f=sawtooth(t, T) % SAWTOOTH Sawtooth Waveform Generation. % SAWTOOTH(t, T) computes values of a sawtooth having a % period T at the values in t. % % Used in Fourier series examples. f=10*rem(t, T)/T; 现在求近似富里哀级数的系数。 >>T= .2; % desired period >>N=25; % number of harmonics >>Fn=fsfind(‘ sawtooth ‘, T, N, T); % compute Fourier series coefficients 因为用FFT求近似系数,所以得到的系数并不精确。不过,如果有足够多的系数,特别是如果时间函数本身,或者其一阶,或者二阶导数连续(锯齿信号不连续),误差就比较小。如果需要更高的精度,可调用积分程序来计算每一个富里哀级数系数的积分关系式。后一种方法需要很长的时间,因为FFT一次要近似所有的系数。不过,尽管有误差,对于许多应用,FTT的近似结果已足够准确了。 用函数fseval可实现富里哀级数的计算。 function y=fseval(kn, t, wo) % FSEVAL Fourier Series Function Evaluation. % FSEVAL(Kn, t, Wo) computes values of a real valued function given % its complex exponential Fourier series coefficients Kn, at the % points given in t where the fundamental frequency is Wo rad/s. % K contains the Fourier coefficentsin ascending order: % Kn = [k k ... k ... k k] % -N -N+1 0 N-1 N % if Wo is not given, Wo=1 ia assumed. % Note: this function creates a matrix of size: % rows = length(t) and columns = (length(K)-1)/2 % Copyrigth (c) 1996 by Prentice-Hall,Inc. if nargin==2, wo=1; end nk=length(kn); if rem(nk-1, 2) | (nk==1) erro(‘ Number of element in K must be odd and greater than 1 ‘) end n=0.5*(nk-1); % highest harmonic nwo=wo*(1 : n); % harmonic frequencies ko=kn(n+1); % average value kn=kn(n+2 : nk) ‘; % positive frequency coefs y=ko+2*(real(exp(j*t( : )*nwo)*kn))' ; 遵照所需的语法,我们得到: >>n=-N : N; % harmonic index >>Fna=j*5 ./ (pi*n); % actual Fourier series coefficients >>Fna(N+1)=5; % poke in average value >>t=linspace(0 , .4); % time points to evaluate functions >>wo=2*pi/T; % fundamental frequency >>f=fseval(Fn, t, wo); % evaluate approximated Fourier series >>fa=fseval(Fna, t, wo); % evaluate actual Fourier series >>plot(t, f, t, fa) % plot results for comparison  图14.3 富里哀级数的结果比较 注意上述两个富里哀级数给出了相似的结果。从图14.3中可知,在锯齿信号的不连续点的周围出现了明显的吉布现象波纹。用函数fswindow,将一个窗口加于富里哀级数系数,就能减少这种波纹。 >>help fswindow FSWINDOW Generate Window Functions. FSWINDOW(N, TYPE) creates a window vector of type TYPE having a length equal to the scale N. FSWINDOW(X, TYPE) creates a window vector of type TYPE having a length and orientation the same as the vector X. FSWINDOW(X, TYPE, alpha) porvides a parameter alpha as required for some window types. FSWINDOW with no input arguments returns a string matrix whose i-th row is the i-th TYPE given below. TYPE is a string designating the window type desired: 'rec' = Rectangular or Boxcar 'tri' = Triangular or Bartlet 'han' = Hann or hanning 'ham' = Hamming 'bla' = blackman common coefs. 'blx' = Blackman exact coefs. 'rie' = Rieman {sin(x)/x} 'tuk' = Tukey, 0<alpha<1; alpha = 0.5 is default 'poi' = Poisson, 0<alpha<inf; alpha = 1 is default 'cau' = Cauchy, 1<alpha<inf; alpha = 1 is default 'gau' = Gaussian, 1<alpha<inf; alpha = 1 is default Reference: F. J. Harris,”On the Use of Windows for Harmonic Analysis with the Discrete Fourier Transform,” IEEE Proc. Vol. 66, no. 1, Jan. 1978, pp.51-83 >>Fnh=Fn.*fswindow(Fn, ‘ han ‘); % apply Hanning window >>f=fseval(Fnh, t, wo); % evaluate windowed FS >>plot(t, f) % plot results  图14.4 用汉宁窗口滤波后的富里哀级数 因为乘以窗口函数,改变了富里哀级数的系数,所以,现在图形(见图14.4)看起来好多了。 接下来,让我们把锯齿波加到传递函数为的线性系统中。 函数fsresp专门解决此类问题。 function y=fsresp(num, den, un, wo) % FSRESP Fourier Series Linear System Response % FSRESP(N, D, Un, Wo) returns the comples exponential FS of the % output of a linear system when the input is given by a FS % N and D are the numerator and denominator coefficients % respectively of the system transfer function. % Un is the complex exponential Fourier Series of the system input. % Wo is the fundamental frequency associated with the input. % Copyright (c) 1996 by Prentice-Hall, Inc. if nargin<4, wo=1; end N=(length(un)-1)/2; % highest harmonic jnWo=sqrt(-1)*(-N : N)*wo; % frequencies of harmonics y=(polyval(num, jnWo) ./ polyval(den, jnWo)).*un; % output is Yn=H(jnWo)*Un 运用fsresp,我们得到: >>Yn=fsresp(60, [1 60], Fn, wo); % find response coefficients >>y=fseval(Yn, t, wo); % evaluate output >>plot(t, f, f, y) % plot system input and output 结果见图14.5。  图14.5 线性系统的输入输出曲线 作为最后一个例子,运用fsform把锯齿信号的富里哀级数转换为三角形式。 function [a, b, ao]=fsform(c, d, co) % FSFORM Fourier Series Format Conversion % KN=FSFORM(An, Bn, Ao) converts the trigonometric FS with % An being the CSOINE and Bn being the SINE coefficients to % the complex exponential FS with coefficients Kn. % Ao is the DC component and An,Bn and Ao are assumed to be real. % % [Kni]=FSFORM(An, Bn, Ao) returns the index vector i that % identifies the harmonic number of each element of Kn. % % [An, Bn, Ao]=FSFORM(Kn) does the reverse format conversion. % Copyrigth (c) 1996 by Prentice-Hall,Inc. nc=length(c); if nargin==1 % complex exp -> trig form if rem(nc-1, 2)|(nc==1) error(‘ Number of elements in K must bo odd and greater than 1 ‘) end nn=(nc+3)/2; a=2*real(c(nn : nc)); b=-2*imag(c(nn : nc)); ao=real(c(nn-1)); elseif nargin==3 % trig from -> complex exp form nd=length(d); if nc~=nd; error(‘ A and B must be the same length ‘) end a=0.5*(c-sqrt(-1)*d); a=[conj(a(nc : -1 : 1)) co(1) a]; b=-nc : nc; else error(‘ Improper number of input arguments. ‘) end >>[An, Bn, Ao]=fsform(Fn) An = Columns 1 through 7 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 Columns 8 through 14 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 Columns 15 through 21 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 -0.2000 Columns 22 through 25 -0.2000 -0.2000 -0.2000 0 Bn = Columns 1 through 7 -3.1789 -1.5832 -1.0484 -0.7789 -0.6155 -0.5051 -0.4250 Columns 8 through 14 -0.3638 -0.3151 -0.2753 -0.2418 -0.2130 -0.1878 -0.1655 Columns 15 through 21 -0.1453 -0.1269 -0.1100 -0.0941 -0.0792 -0.0650 -0.0514 Columns 22 through 25 -0.0382 -0.0253 -0.0126 0 Ao = 4.9000 忽略平均值,这个锯齿应展示了奇对称特性,所有An系数应当为0。显然,它们不为0。发生这种情况的原因是因为我们使用了FFT来近似系数。此外,Ao平均值有一点儿误差,它的值应当是5.0。 14.3 小结 下面的两个表总结了本章所讨论的函数。 表14.3 信号处理函数  conv 卷积  conv2 2维卷积  fft 快速富里哀变换  fft2 2维快速富里哀变换  ifft 逆快速富里哀变换  ifft2 2维逆快速富里哀变换  filter 离散时间滤波器  filter2 2维离散时间滤波器  abs 幅值  angle 四个象限的相角  unwrap 在360°边界清除相角突变  fftshift 把FFT结果平移到负频率上  nextpow2 2的下一个较高幂次   表14.4 精通MATLAB的富里哀级数函数  fsderiv(Kn,Wo) 富里哀级数的微分  fseval(Kn,t,Wo) 计算富里哀级数  fsfind(‘ fname ‘,T,N) 寻找时间函数的富里哀级数的系数  [An,Bn,Ao]=fsform(Kn) Kn=fsform(An,Bn,Ao) 富里哀级数不同形式之间的转换  fsharm(Kn,i) 提取特殊的富里哀级数的谐波  fsmsv(Kn) 计算信号的均方根值  fsresize(Kn,N) 重新调整富里哀级数的系数向量的大小  fsresp(Num,Den,Un,Wo) 线性系统对输入富里哀级数Un的富里哀级数响应  fsround(Kn) 设置次要的富里哀级数的系数为0  fswindow(N, ‘ type ‘) fswindow(Kn, ‘ type ‘) 产生一个窗口的函数,使吉布(Gibb)现象极小   第15章 低级文件I/O 对于大多数用户,MATLAB函数load和save为装载和存储数据提供了足够的工具。利用以扩展名为.mat结尾的文件名,load和save假定数据是以与平台无关的二进制格式保存,或者用称之为flat的简单的ASCII文件格式保存。当flat ASCII或.mat这两种格式还不够时,MATLAB提供了基于C语言实现的低级文件I/O函数。用这些低级文件I/O函数,MATLAB可以读写你所知道的任意文件格式。例如,知道电子文件或数据库程序所使用的格式,就可以把这些数据读进MATLAB的矩阵中去。类似地,也可以创建电子文件或数据库文件。 MATLAT中这种基本的低级文件I/O命令如下: 表15.1 MATLAB低级文件I/O函数  fclose: 关闭文件  feof: 测试文件结束  ferror: 查询文件I/O的错误状态  fgetl: 读文件的行,忽略回行符  fgets: 读文件的行,包括回行符  fpoen: 打开文件  fprintf: 把格式化数据写到文件或屏幕上  fread: 从文件中读二进制数据  frewind: 返回到文件开始  fscanf: 从文件中读格式化数据  fseek: 设置文件位置指示符  ftell: 获取文件位置指示符  fwrite: 把二进制数据写到文件里   除了这些函数外,所具有的MATLAB版本可能为一个或多个公用软件包提供读写文件的特定函数M文件。有关这些函数的进一步的信息,请使用在线帮助:>>help iofun 第16章 调试工具 在开发函数M文件过程中,不可避免地出现错误,即故障。MATLAB提供了很多函数和方法,帮助调试函数。 在MATLAB表达式中,有两类错误:语法错误和运行错误。当MATLAB计算一个表达式的值或一个函数被编译到内存时会发现语法错误。一旦发现语法错误,MATLAB立即标志这些错误,并提供有关所遇到的错误类型,以及发生错误处M文件的行数。给定这些反馈信息,就很容易纠正这些错误。 而另一方面,即使MATLAB标志了运行错误,但找出错误一般比较困难。当发现运行错误时,MATLAB把控制权返回给命令窗口和MATLAB的工作空间。失去了对发生错误的函数空间的访问权,因此,用户不能询问函数工作空间中的内容排除问题。 根据作者的经验,当一些操作结果导致空矩阵或NaNs时,最容易发生运行错误。所有有关NaNs的操作都返回NaNs值。因此,如果有可能出现NaNs结果,则当出现NaNs时,最好运用逻辑函数isnan来执行一些缺省操作。因为空矩阵为零维,所以对空矩阵寻址常常导致错误。函数find表示了可产生空矩阵结果的一般情况。如果函数find的空矩阵输出用于索引其它数组,所返回的值也将是空的。这样,空矩阵具有传播性质。例如: >>x=pi*(1 : 4) % example data >>i=find(x>20) % use find function >>y=2*x(i) % propagate the empty matrix 很清楚,当希望y具有有限维数和值时,可能发生运行错误。当执行一个操作或使用可返回空结果的函数时,逻辑函数isempty有利于为空矩阵定义一个缺省值,这样避免运行错误。 有几种调试函数M文件的方法。对于简单的问题,可直接使用下列的方法组合: 1、去掉文件中所选择的行的分号,以便中间结果显示在命令窗口中。 2、在文件中加入显示感兴趣的变量的语句 3、把keyboard命令放在文件中所选择的地方,给键盘暂时控制权。这样,可以查询函数空间并按需要改变其值。 4、在M文件开始,在function语句前加上%,将函数M文件改变为脚本M文件。当MATLAB执行该脚本M文件时,该空间就是MATLAB工作空间。这样,当发生错误时可以询问。 当M文件大,递归调用或者多次嵌套(即调用其它M文件函数,被调用M文件函数又调用其它M文件函数,等等)时,用MATLAB的调试函数会更方便。与上述所列方法相反,这些调试函数不要求将有问题的M文件进行编辑。表16.1所给出的这些函数类似于其它高级编程语言中所提供的函数。有关进一步的信息,以及它们的使用实例,参阅《MATLAB用户指南》。 表16.1 MATLAB调试函数  dbclear: 取消断点  dbcont: 在断点后恢复运行  dbdown: 工作空间下移  dbquit: 退出调试模式  dbstack: 列出谁调用谁  dbstatus: 列出所用的断点  dbstep: 执行一行或多行  dbstop: 设置断点  dbtype: 列出带行号的M文件  dbup: 工作空间上移