单片机频率测量原理
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
单片机频率测量原理
单片机应用系统中,经常要对一个连续的脉冲波频率进行测量。在实际应用中,对于转速,位移、速度、流量等物理量的测量,一般也是由传感器转换成脉冲电信号,采用测量频率的手段实现。
使用单片机测量频率或周期,通常是利用单片机的定时计数器来完成的,测量的基本方法和原理有两种:
测频法:在限定的时间内(如1秒钟)检测脉冲的个数。
测周法:测试限定的脉冲个数之间的时间。
这两种方法尽管原理是相同的,但在实际使用时,需要根据待测频率的范围、系统的时钟周期、计数器的长度、以及所要求的测量精度等因素进行全面和具体的考虑,寻找和设计出适合具体要求的测量方法。
在具体频率的测量中,需要考虑和注意的因素有以下几点。
ü系统的时钟。首先测量频率的系统时钟本身精度要高,因为不管是限定测量时间还是测量限定脉冲个数的周期,其基本的时间基准是系统本身时钟产生的。其次是系统时钟的频率值,因为系统时钟频率越高,能够实现频率测量的精度也越高。因此使用A VR测量频率时,建议使用由外部晶体组成的系统的振荡电路,不使用其内部的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
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 interrupt service 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码转换和送入显示