西部热线 | 助力西部开发,关注西部民生! |
adtop
adtop01
当前位置: 西部热线 > 科技

单片机频率测量原理

作者:余梓阳    栏目:科技    来源:西部热线    发布时间:2016-09-01 17:19

单片机应用系统中,常常要对一个持续的脉冲波频率进行丈量。在实际应用中,关于转速,位移、速度、流量等物理量的丈量,普通也是由 传感器转换成脉冲电信号,采取丈量频率的手段完成。

应用单片机丈量频率或周期,通常是利用单片机的定时计数器来完成的,丈量的基本办法和原理有两种:

测频法:在限制的时间内(如1秒钟)检测脉冲的个数。

测周法:测试限制的脉冲个数之间的时间。

这两种办法虽然原理是一样的,但在实际应用时,需求依据待测频率的范围、系统的时钟周期、计数器的长度、和所请求的丈量精度等因素进行整体和详细的考虑,寻觅和设计出适合详细请求的丈量办法。

在详细频率的丈量中,需求考虑和留意的因素有以下几点。

ü 系统的时钟。首先丈量频率的系统时钟自己精度要高,缘由是不顾是限制丈量时间还是丈量限制脉冲个数的周期,其基本的时间基准是系统自己时钟产生的。其次是系统时钟的频率值,缘由是系统时钟频率越高,能够完成频率丈量的精度也越高。所以应用AVR丈量频率时,建议应用由外部晶体构成的系统的振荡电路,不应用其内部的RC振荡源,同时尽也许应用频率比较高的系统时钟。

ü 所应用定时计数器的位数。丈量频率要应用定时计数器,定时计数器的位数越长,能够产生的限制时间越长,或在限制时间里记载的脉冲个数越多,所以也提升了频率丈量的精度。所以对频率丈量精度有必定请求时,尽也许采取16位的定时计数器。

ü 被测频率的范围。频率丈量需求依据被测频率的范围选择丈量的方法。当被测频率的范围比较低时,最好采取测周期的办法丈量频率。而被测频率比较高时,应用测频法比较适合。需求留意的是,被测频率的最高值普通不能够超出测频MCU系统时钟频率的1/2,缘由是当被测频率高于MCU时钟1/2后,MCU常常不能够准确检测被测脉冲的电平变化了。

除过以上三个因素外,还要考虑频率丈量的频度(每秒内丈量的次数),如何与系统中其它任务处置之间的调和工作等。频率丈量精度请求高时,还应当考虑其它中止和中止呼应时间的影响,甚至需求在软件中考虑采取屡次丈量取平均的算法等。

采取测频法的频率计设计与完成

1) 硬件电路

硬件电路的显示部分,PA口为8个LED数码管的段输出,PC口控制8个LED数码管的位扫描。应用T/C0对被测信号输入的脉冲个数进行计数,被测频率信号由PB0(T0)输入。

2) 软件设计

我们首先给出系统程序,然后做必要的说明。

/*********************************************

File name : demo_11_1.c

Chip type : ATmega16

Program type : Application

Clock frequency : 4.000000 MHz

Memory model : Small

External SRAM size : 0

Data Stack size : 256

*********************************************/

#include <mega16.h>

flash char led_7[10]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};

flash char position[8]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe};

char dis_buff[8]; // 显示缓冲区,寄存要显示的8个字符的段码值

char posit;

bit time_1ms_ok,display_ok=0;

char time0_old,time0_new,freq_time;

unsigned int freq;

void display(void) // 8位LED数码管动态扫描函数

{

PORTC = 0xff;

PORTA = led_7[dis_buff[posit]];

if (posit==5) PORTA = PORTA | 0x80;

PORTC = position[posit];

if (++posit >=8 ) posit = 0;

}

// Timer 2 output compare interruptservice routine

interrupt [TIM2_COMP] void timer2_comp_isr(void)

{

time0_new = TCNT0; // 1ms到,记载如今T/C0的计数值

time_1ms_ok = 1;

display_ok = ~display_ok;

if (display_ok) display();

}

void freq_to_disbuff(void) // 将频率值转化为BCD码并送入显示缓冲区

{

char i,j=7;

for (i=0;i<=4;i++)

{

dis_buff[j-i] = freq % 10;

freq = freq / 10;

}

dis_buff[2] = freq;

}

void main(void)

{

char i;

DDRA=0xFF; // LED数码管驱动

DDRC=0xFF;

// T/C0初始化,外部计数方法

TCCR0=0x06; // 外部T0脚降低沿触发计数,普通形式

TCNT0=0x00;

OCR0=0x00;

// T/C2初始化

TCCR2=0x0B; // 内部时钟,32分频(4M/32=125KHz),CTC形式

TCNT2=0x00;

OCR2=0x7C; // OCR2 = 0x7C(124),(124+1)/125=1ms

TIMSK=0x80; // 许可T/C2比较配套中止

for (i=0;i<=7;i++) dis_buff[i] = 0;

time0_old = 0;

#asm("sei") // 开放全局中止

while (1)

{

if (time_1ms_ok)

{ // 累计T/C0的计数值

if (time0_new >= time0_old) freq = freq + (time0_new - time0_old);

else freq = freq + (256 - time0_old + time0_new);

time0_old = time0_new;

if (++freq_time >= 100)

{

freq_time = 0; // 100ms到,

freq_to_disbuff(); // 将100ms内的脉冲计数值送显示

freq = 0;

}

time_1ms_ok = 0;

}

};

}

程序中LED扫描形式函数desplay(),和脉冲计数值转换成BCD码并送显示缓冲区函数freq_to_disbuff()比较简单,请读者自己分析。

在该程序中,应用了两个定时计数器。T/C0工作在计数器方法,对外部T0引脚输入的脉冲信号计数(降低沿触发)。T/C2工作在CTC方法,每隔1ms中止一次,该定不时间即作为LED的显示扫描,同时也是限制时间的基时。每次T/C2的中止中,都首先记载下T/C0寄存器TCNT0如今的计数值,所以前后两次TCNT0的差值(time0_new – time0_old)或(256 - time0_old + time0_new)就是1ms时间内T0脚输入的脉冲个数。为了提升丈量精度,程序对100个1ms的脉冲个数进行了累计(在变量freq中),即已知限制的时间为100ms。

读者还应当留意频率的持续丈量与LED扫描、BCD码转换之间的调和问题。T/C2中止距离为1ms,所以在1ms时间内,程序必须将脉冲个数进行的累计、BCD码转换和送入显示缓冲区,和LED的扫描工作完成掉,不然就会影响到下一次中止到来后的处置。

在本实例的T/C2中止中,应用了display_ok标记,将LED扫描分配在奇数ms(1、3、5、7、……),而将1ms的TCNT0差值计算、积累和转换等处置放在主程序中完成。另外因为计算量大的BCD码转换是在偶数ms(100ms)处置,所以程序中LED的扫描处置和BCD码转换处置不会同时进行(不会在两次中止距离的1ms内同时处置LED扫描和BCD码转换),这就保障了在下一次中止达到时,前一次的处置已经所有完成,使频率的持续丈量不受影响。

该实例程序的性能和指标为(假定系统时钟没有误差 = 4MHz):

ü 频率丈量绝对误差:±10Hz。因为限制的时间为100ms,并且T/C0的计数值有±1的误差,换算成频率为±10Hz。

ü 被测最高频率值:255KHz。因为T/C0的长度8位,所以在1ms中,TO输入的脉冲个数应小于255个,大于255后形成T/C0的自动清另,丧失脉冲个数。

ü 丈量频度:10次/秒。限制的时间为100ms,持续丈量,所认为10次/秒。

ü 应用资源:两个定时器,一个中止。

原文链接:http://www.eeworld.com.cn/mcu/article_2016090128993.html

adl03
adr1
adr2