IIR滤波器软件实现(Matlab+C++)
使用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为各节的增益,此项为了稳定各节,稳定信号大小
接着对照头文件,以下为头文件主要部分的一段截取:
#define MWSPT_NSEC 41
int NL[MWSPT_NSEC] = { ,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,
,,,,, };
double NUM[MWSPT_NSEC][] = {
{
0.3416221864767, ,
},
{
, ,
},
{
0.3180955154747, ,
},
{
, ,
},......
MWSPT_NSEC为滤波器阶数,具体的节数在头文件开头的注释中
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完全滤除。
上图为原信号,下图为滤波后信号。
实际测试发现符合设计要求,而且在过渡带信号也基本完全衰减。
测试用C++程序
void main()
{
const int N = ;
int i,j;
double xn[N];
double w[][];
double yn[N]; for (i = ; i < ; i++)//初始化
{
for (j = ; j < ; j++)
w[i][j] = ;
} for (i = ; i < N; i++)
{
xn[i] = sin( * 3.1416 * / * i)+ sin( * 3.1416 * / * i)+ sin( * 3.1416 * / * i); yn[i]=iir(&DEN[][], &NUM[][], &w[][],xn[i], ); } ofstream SaveFile_a("xn.txt");
for (i = ; i<N; i++)
SaveFile_a << " " << xn[i] << endl;
SaveFile_a.close(); ofstream SaveFile_b("yn.txt");
for (i = ; i<N; i++)
SaveFile_b << " " << yn[i] << endl;
SaveFile_a.close();
} 分析用Matlab程序
xn1=fopen('xn.txt','r');
[xn,count]=fscanf(xn1,'%f');
fclose(xn1); N = length(xn);%求取抽样点数
xn_f = fft(xn);%对信号进行傅里叶变换
xn_f=abs(xn_f(:N/));
f = /N*(:N/-); subplot();
stem(f,abs(xn_f));
xlabel('Frequency / (s)');ylabel('Amplitude'); title('原信号频谱');
grid; yn1=fopen('yn.txt','r');
[yn,count]=fscanf(yn1,'%f');
fclose(yn1); yn_f = fft(yn);%对信号进行傅里叶变换
yn_f=abs(yn_f(:N/));
subplot();
stem(f,abs(yn_f));
xlabel('Frequency / (s)');ylabel('Amplitude');
title('滤波后信号频谱');
grid;
IIR滤波器软件实现(Matlab+C++)的更多相关文章
- IIR滤波器设计(调用MATLAB IIR函数来实现)
转载请注明文章来源 – http://blog.csdn.net/v_hyx ,请勿用于任何商业用途 对于滤波器设计,以前虽然学过相关的理论(现代数字信号处理和DSP设计),但一直不求 ...
- IIR滤波器和FIR滤波器的区别与联系zz
-------------------------------------------------------------------------------------------------- ...
- 数字信号处理实验(五)——IIR滤波器的设计
一.使用自编函数设计IIR滤波器 1.冲激响应法 (1)注给出的数字滤波器指标先化成模拟指标 (2)设计出模拟滤波器: (3)使用冲激响应法转化成数字滤波器 (4)一个demo clear all; ...
- 基于FPGA的IIR滤波器
基于FPGA的IIR滤波器 by方阳 版权声明:本文为博主原创文章,转载请指明转载地址 ...
- 杂项-数学软件:MATLAB
ylbtech-杂项-数学软件:MATLAB MATLAB是美国MathWorks公司出品的商业数学软件,用于算法开发.数据可视化.数据分析以及数值计算的高级技术计算语言和交互式环境,主要包括MATL ...
- 手把手教系列之IIR滤波器设计
[导读]:在嵌入式系统中经常需要采集模拟信号,采集模拟信号的信号链中难免引入干扰,那么如何滤除干扰呢?今天就来个一步一步描述如何设计部署一个IIR滤波器到你的系统. 何为IIR滤波器? 无限冲激响应( ...
- FIR滤波器和IIR滤波器的区别
数字滤波器广泛应用于硬件电路设计,在离散系统中尤为常见,一般可以分为FIR滤波器和IIR滤波器,那么他们有什么区别和联系呢. FIR滤波器 定义: FIR滤波器是有限长单位冲激响应滤波器,又称为非递归 ...
- FIR滤波器与IIR滤波器
FIR(Finite Impulse Response)滤波器 有限长单位冲激响应滤波器,又称为非递归型滤波器 特点: FIR滤波器的最主要的特点是没有反馈回路,稳定性强,故不存在不稳定的问题: FI ...
- [math][mathematica] archlinux 下 mathematica 的安装 (科学计算软件 mathematica/matlab/sagemath)
ONLINE: http://www.wolframalpha.com/ GPL: segamath: http://www.sagemath.org/ famous and not free: ma ...
随机推荐
- Java-IO之InputStreamReader和OutputStreamWriter
InputStreamReader和OutputStreamWriter是字节流通向字符流的桥梁.它使用指定的差染色体读写字节并将其解码为字符.InputStreamReader的作用是将字节输入流转 ...
- Struts2进阶(一)运行原理及搭建步骤
Struts2进阶(一)运行原理 Struts2框架 Struts2框架搭建步骤 致力于web服务,不可避免的涉及到编程实现部分功能.考虑使用到SSH框架中的Struts2.本篇文章只为深入理解Str ...
- 最简单的基于FFMPEG+SDL的音频播放器 ver2 (采用SDL2.0)
===================================================== 最简单的基于FFmpeg的音频播放器系列文章列表: <最简单的基于FFMPEG+SDL ...
- pig 的chararry类型不能用比较运算符comparison operator
pig 的chararry类型可能是按字段,逐个字段进行比较. element_id 是chararray类型, 语句: no_app_category_mapping = filter no_ele ...
- Android studio中的一次编译报错’Error:Execution failed for task ':app:transformClassesWithDexForDebug‘,困扰了两天
先说下背景:随着各种第三方框架的使用,studio在编译打包成apk时,在dex如果发现有相同的jar包,不能创建dalvik虚拟机.一个apk,就是一个运行在linux上的一个虚拟机. 上图就是一直 ...
- JavaScript压缩工具JSA使用介绍
JavaScript压缩工具JSA使用介绍 JSA绝对是我使用过的JS压缩工具中最上乘的一个.认识它是从ligerUI开始.在ligerUI的QQ讨论组里,大神--ligerUI的作者告诉我他的lig ...
- 我眼中的Linux设备树(六 memory&chosen节点)
六 memory&chosen节点根节点那一节我们说过,最简单的设备树也必须包含cpus节点和memory节点.memory节点用来描述硬件内存布局的.如果有多块内存,既可以通过多个memor ...
- driver: Linux设备模型之input子系统详解
本节从整体上讲解了输入子系统的框架结构.有助于读者从整体上认识linux的输入子系统.在陷入代码分析的过程中,通过本节的知识能够找准方向,明白原理. 本节重点: 输入子系统的框架结构 各层对应内核中的 ...
- Linux 学习笔记_12_Windows与Linux文件共享服务_1.1_--Samba(下)Samba经典应用案例
五.[Samba应用案例二] 设置Samba共享目录/software,允许用户jack和mary可以通过Windows客户端访问,并具有读写权限. 1.创建目录/software,添加用户jack, ...
- SpriteBuilder中节点位置类型为百分比时不能定位的解决
Ball.ccb类型是Node,其中有个子节点为Color Node,其中物理使能. MainScene.ccb中加入一个物理节点,将Ball.ccb拖入其中,成为该物理节点的孩子,这时出现了一个&q ...