使用C++来写一个IIR滤波器

我们首先要在MATLAB中设计一个IIR滤波器,并生成一个头文件,这个头文件中反映了IIR滤波器的频率响应特性


理论支持

IIR滤波叫做递归滤波器,它是一种具有反馈的滤波器。当阶数较大时一般采取多个二阶节滤波进行串联,这样可以提高系统稳定性。

一个二阶节系数规律如图所示:

可以写出第K个二阶节的差分方程

N个二阶节的级联结构如下图所示:

根据二阶节图,把前一级的输出作为后一级的输入,就可以通过软件实现IIR数字滤波的功能。


使用Matlab生成头文件

首先打开MATLAB中Filter Design & Analysis Tool

这里我们先设计一个低通滤波器

Fs代表采样频率,采样频率必须大于原信号最高频率的两倍,

否则会产生频谱混叠。

Fpass为通带频率,Fstop为阻带截止频率

这些参数设置好就可以点击Design Filter

生成的是一个二阶节滤波组合,一共有31阶,也就是多个二阶滤波器的组合

接着在Target选项中生成C Header File

Numerator为分子系数数组的命名,Numerator length为分子系数数组的长度,

Denominator为分母。


对生成头文件进行分析

以下以Fpass为10K,Fstop为12K的低通滤波器举栵

在使用头文件前需要根据情况将Matlab的数据类型转换为C++支持的数据类型,这里我们使用double类型

在分析头文件前先看下Matlab提供的第一节滤波参数

以第一个二阶节的数据举例:

  • Numerator: 1  2  1
  • Denominator: 1  -0.55930961405944157  0.92579835996619642
  • Gain:0.34162218647668868

Numerator为分子的系数,分别为b0,b1,b2

Denominator为分母的系数,分别为a0,a1,a2.

Gain为各节的增益,此项为了稳定各节,稳定信号大小

接着对照头文件,以下为头文件主要部分的一段截取:

  1. #define MWSPT_NSEC 41
  2. int NL[MWSPT_NSEC] = { ,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,
  3. ,,,,, };
  4. double NUM[MWSPT_NSEC][] = {
  5. {
  6. 0.3416221864767, ,
  7. },
  8. {
  9. , ,
  10. },
  11. {
  12. 0.3180955154747, ,
  13. },
  14. {
  15. , ,
  16. },......
  1. MWSPT_NSEC为滤波器阶数,具体的节数在头文件开头的注释中
  1. NL[MWSPT_NSEC]这个数组定义了NUM[MWSPT_NSEC][3]数组每一行的有用数据个数(可以不用)

在NUM[MWSPT_NSEC][3]数组(分子参数)奇数行第一项都为增益项,偶数行为3个系数,分别为b0,b1,b2。

由此可以找出规律,定义K为目前所在的阶数,p为数组的首指针,则,每一节的增益项为(p+6*K),第一个系数为(p+3+6*K),

第二个系数为(p+3+6*K+1),第三个系数为(p+3+6*K+2)。


C++编程实现

在软件设计的过程中,每个二阶节的延迟变量只取 和 , 作为中间变量在过程中直接赋给 。这是因为对于下一个输入数据n+1的延迟变量即为上一个输入数据的 和 ,采用这种方式进行设计,可以节省寄存器的空间。

为了提高处理速度,程序中需要使用指针进行参数传递,特别注意二维数组的首地址传递方式为&a[0][0]->double* a。

滤波函数

double iir(double *a, double *b,double* w, double xin, int N_IIR)

{

int k;

double temp = xin;

for (k = 0; k<N_IIR; k++)

{

*(w+k*3) = temp - *(a + 3+6 * k  + 1) *(*(w + k * 3+1)) - *(a + 3 + 6 * k + 2) *(*(w + k * 3+2));

//这里temp为本二阶节的输入,也是上一个二阶节的输出

temp = *(b + 3 + 6 * k )* (*(w + k * 3)) + *(b + 3 + 6 * k + 1) * (*(w + k * 3+1)) + *(b + 3+6 * k + 2)* (*(w + k * 3+2));

//这里temp为本二阶节的输出,也是下一个二阶节的输入

*(w + k * 3 + 2) = *(w + k * 3 + 1);

*(w + k * 3 + 1) = *(w + k * 3);

temp = temp*(*(b + 6 * k));//放大倍数,稳定信号

}

return temp;

}


实际测试

测试Fpass为10K,Fstop为12K的低通滤波器

在程序中输入三个频率为2K,11K,20K的信号,理论上2k完全通过,11k部分衰减,20K完全滤除。

上图为原信号,下图为滤波后信号。

实际测试发现符合设计要求,而且在过渡带信号也基本完全衰减。

  1. 测试用C++程序
  2. void main()
  3. {
  4. const int N = ;
  5. int i,j;
  6. double xn[N];
  7. double w[][];
  8. double yn[N];
  9.  
  10. for (i = ; i < ; i++)//初始化
  11. {
  12. for (j = ; j < ; j++)
  13. w[i][j] = ;
  14. }
  15.  
  16. for (i = ; i < N; i++)
  17. {
  18. xn[i] = sin( * 3.1416 * / * i)+ sin( * 3.1416 * / * i)+ sin( * 3.1416 * / * i);
  19.  
  20. yn[i]=iir(&DEN[][], &NUM[][], &w[][],xn[i], );
  21.  
  22. }
  23.  
  24. ofstream SaveFile_a("xn.txt");
  25. for (i = ; i<N; i++)
  26. SaveFile_a << " " << xn[i] << endl;
  27. SaveFile_a.close();
  28.  
  29. ofstream SaveFile_b("yn.txt");
  30. for (i = ; i<N; i++)
  31. SaveFile_b << " " << yn[i] << endl;
  32. SaveFile_a.close();
  33. }
  34.  
  35. 分析用Matlab程序
  36. xn1=fopen('xn.txt','r');
  37. [xn,count]=fscanf(xn1,'%f');
  38. fclose(xn1);
  39.  
  40. N = length(xn);%求取抽样点数
  41. xn_f = fft(xn);%对信号进行傅里叶变换
  42. xn_f=abs(xn_f(:N/));
  43. f = /N*(:N/-);
  44.  
  45. subplot();
  46. stem(f,abs(xn_f));
  47. xlabel('Frequency / (s)');ylabel('Amplitude');
  48.  
  49. title('原信号频谱');
  50. grid;
  51.  
  52. yn1=fopen('yn.txt','r');
  53. [yn,count]=fscanf(yn1,'%f');
  54. fclose(yn1);
  55.  
  56. yn_f = fft(yn);%对信号进行傅里叶变换
  57. yn_f=abs(yn_f(:N/));
  58. subplot();
  59. stem(f,abs(yn_f));
  60. xlabel('Frequency / (s)');ylabel('Amplitude');
  61. title('滤波后信号频谱');
  62. grid;

IIR滤波器软件实现(Matlab+C++)的更多相关文章

  1. IIR滤波器设计(调用MATLAB IIR函数来实现)

    转载请注明文章来源 – http://blog.csdn.net/v_hyx ,请勿用于任何商业用途         对于滤波器设计,以前虽然学过相关的理论(现代数字信号处理和DSP设计),但一直不求 ...

  2. IIR滤波器和FIR滤波器的区别与联系zz

      -------------------------------------------------------------------------------------------------- ...

  3. 数字信号处理实验(五)——IIR滤波器的设计

    一.使用自编函数设计IIR滤波器 1.冲激响应法 (1)注给出的数字滤波器指标先化成模拟指标 (2)设计出模拟滤波器: (3)使用冲激响应法转化成数字滤波器 (4)一个demo clear all; ...

  4. 基于FPGA的IIR滤波器

    基于FPGA的IIR滤波器                                                         by方阳 版权声明:本文为博主原创文章,转载请指明转载地址 ...

  5. 杂项-数学软件:MATLAB

    ylbtech-杂项-数学软件:MATLAB MATLAB是美国MathWorks公司出品的商业数学软件,用于算法开发.数据可视化.数据分析以及数值计算的高级技术计算语言和交互式环境,主要包括MATL ...

  6. 手把手教系列之IIR滤波器设计

    [导读]:在嵌入式系统中经常需要采集模拟信号,采集模拟信号的信号链中难免引入干扰,那么如何滤除干扰呢?今天就来个一步一步描述如何设计部署一个IIR滤波器到你的系统. 何为IIR滤波器? 无限冲激响应( ...

  7. FIR滤波器和IIR滤波器的区别

    数字滤波器广泛应用于硬件电路设计,在离散系统中尤为常见,一般可以分为FIR滤波器和IIR滤波器,那么他们有什么区别和联系呢. FIR滤波器 定义: FIR滤波器是有限长单位冲激响应滤波器,又称为非递归 ...

  8. FIR滤波器与IIR滤波器

    FIR(Finite Impulse Response)滤波器 有限长单位冲激响应滤波器,又称为非递归型滤波器 特点: FIR滤波器的最主要的特点是没有反馈回路,稳定性强,故不存在不稳定的问题: FI ...

  9. [math][mathematica] archlinux 下 mathematica 的安装 (科学计算软件 mathematica/matlab/sagemath)

    ONLINE: http://www.wolframalpha.com/ GPL: segamath: http://www.sagemath.org/ famous and not free: ma ...

随机推荐

  1. React Native项目组织结构介绍

    代码组织: 目录结构: . ├── components //组成应用的各个组件 │   ├── Routers.android.js //每个组件若实现不一样,分为android的实现和ios的实现 ...

  2. Spring MVC 入门示例讲解 - howtodoinjava

    在本例中,我们将使用Spring MVC框架构建一个入门级web应用程序.Spring MVC 是Spring框架最重要的的模块之一.它以强大的Spring IoC容器为基础,并充分利用容器的特性来简 ...

  3. Android官方命令深入分析之etc1tool

    etc1tool是一个命令行工具,可以将PNG图像压缩为etc1标准,并且可以进行解压缩. 用法: etc1tool infile [--help | --encode | --encodeNoHea ...

  4. cocos2d-js(二)cocos2d-js的基本语法与类的简介

    基本语法: 1.类的定义 一般类都是集成Scene或者Layer: var myLayer = cc.Layer.extend({类的内容}); 2类内的成员变量与方法: 2.1成员变量的声明: 变量 ...

  5. android的PackageManagerService详解

    PackageManagerService主要是一个包的管理服务,在开机的时候会解析以前保存的一些安装包的相关数据,android运行过程中新安装的apk也会保存到PackageManagerServ ...

  6. 【Android 应用开发】Activity 状态保存 OnSaveInstanceState参数解析

    作者 : 韩曙亮 转载请著名出处 : http://blog.csdn.net/shulianghan/article/details/38297083 一. 相关方法简介 1. 状态保存方法示例 p ...

  7. 【一天一道LeetCode】#75. Sort Colors

    一天一道LeetCode 本系列文章已全部上传至我的github,地址:ZeeCoder's Github 欢迎大家关注我的新浪微博,我的新浪微博 欢迎转载,转载请注明出处 (一)题目 Given a ...

  8. Dynamics CRM2013 Server2012R2下IFD部署遇到There is already a listener on IP endpoint的解决方法

    接上一篇继续Server2012R2的问题,因为自己先在R2上部署的IFD报错后上网查了很多资料,但毕竟R2是新出的CRM2013也是新出的,网上基本还没有相关的问题反馈,基本都是2012以前的系统版 ...

  9. android放大镜效果实现

    概述 我相信很多用过英语应用的同学都看多一个放大镜的效果,就是选中一段文字后,会有一个放大镜,这个究竟怎么实现的呢,我们今天来分析分析. 源码分析 public class ShaderView ex ...

  10. AWR報告詳解

    AWR是Oracle  10g 版本 推出的新特性, 全称叫Automatic Workload Repository-自动负载信息库, AWR 是通过对比两次快照(snapshot)收集到的统计信息 ...