
#include "reg51.h"
#define uchar unsigned char
uchar disp[8]={0,0,0,0,0,0,0,0}
uchar T0count,T1count
void delay(void)
{
uchar i
for(i=250i>0i--)
}
void display()
{
//uchar i,j,k=0x80
uchar dispcode[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}
uchar i,k
k=0x80
for(i=0i<8i++)
{
P2=0
P0=dispcode[disp[i]]
P0=~P0
P2=k
k=k>>1
delay()
}
P2=0
}
void calc()
{
uchar i
long frequency
frequency=(T0count*256+TH0)*256+TL0
for(i=7i>0i--)
{
disp[i]=frequency%10
frequency=frequency/10
}
disp[0]=frequency
}
void init()
{
T0count=0
T1count=0
TH0=0
TL0=0
}
void main()
{
init()
TMOD=0x15
TH1=(65536-5*110592/12)/256
TL1=(65536-5*110592/12)/256%10
ET1=1
ET0=1
EA=1
TR1=1
TR0=1
//以下四句的作用是在P1.0引脚上形成1000Hz的脉冲,用导线连接到P3.4作为测试用,如果是AT89S51,则四句不用。将其中
//高8位和低8位的初始值更改后可输出不同频率的脉冲。
/*
T2MOD=0x2
RCAP2H=245
RCAP2L=74
TR2=1
*/
while(1)
{
display()
}
}
void time0() interrupt 1
{
T0count++
}
void time1() interrupt 3
{
TH1=(65536-5*110592/12)/256
TL1=(65536-5*110592/12)/256%10
if(T1count==19)
{
calc()
init()
}
else T1count++
}
常用51单片机频率计有两种方法,一种方法,定时计脉冲个数,如定时100ms,计数器计值是55,则频率为55x10=550hz,另外一种方法,计单个脉冲的周期,可用外部中断,也可不用外部中断,下降沿到来时开定时器,上升沿或另外一个下降沿到来时关闭定时器,读出定时器的值,若脉冲宽度太宽,定时器可能会溢出, 也可以让定时器中断,定时器计时加上中断溢出次数的时间,就是总时间=周期,显然你用的是第二种方法程序基本没问题,但你的显示程序应该是包括while(1){while(pulse==0).....}
这个大偱环内的,不要另起一个偱环,还是真正的死偱环,跳不出来,结果是复位一次,只能测量一次
这种方法测低频比较准,但也有一个缺点,测低频时,更新很慢,测1HZ脉冲,必须要2秒,1S测量计时,另外几ms处理并显示,剩下的近1S空等
测0.5HZ脉冲,理论上要4秒,所以要求0.5-1S更新一次和测量的频率下限是矛盾的
既然用了中断TF1会自动清0,TF1=0是多余的
还有,你若是液晶屏显示还可以,数码管测低频还存在着刷新问题,长时间的等待,数码管会只亮一个,如果把数码管刷新程序放在等待空闲内的话,测高频时会错过脉冲上升和下降沿,测低频也会,只不过几率小些,所以用数码管的话,也只能用静态显示
或者间断显示也可以,即将显示子程序偱环50-100次(持续1.5-3s),然后再次测量,把while(1) {ledxianshi()}死偱环改成FOR偱环
也可以用第一种方法测,若测试低频的话,就要把定时时间设得很长,测0.1HZ信号,你不定时大于10s,测出来就是0
3. 系统板上硬件连线
(1). 把“单片机系统”区域中的P0.0-P0.7与“动态数码显示”区域中的ABCDEFGH端口用8芯排线连接。
(2). 把“单片机系统”区域中的P2.0-P2.7与“动态数码显示”区域中的S1S2S3S4S5S6S7S8端口用8芯排线连接。
(3). 把“单片机系统”区域中的P3.4(T0)端子用导线连接到“频率产生器”区域中的WAVE端子上。
4. 程序设计内容
(1). 定时/计数器T0和T1的工作方式设置,由图可知,T0是工作在计数状态下,对输入的频率信号进行计数,但对工作在计数状态下的T0,最大计数值为fOSC/24,由于fOSC=12MHz,因此:T0的最大计数频率为250KHz。对于频率的概念就是在一秒只数脉冲的个数,即为频率值。所以T1工作在定时状态下,每定时1秒中到,就停止T0的计数,而从T0的计数单元中读取计数的数值,然后进行数据处理。送到数码管显示出来。
(2). T1工作在定时状态下,最大定时时间为65ms,达不到1秒的定时,所以采用定时50ms,共定时20次,即可完成1秒的定时功能。
5. C语言源程序
#include <AT89X52.H>
unsigned char code dispbit[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}
unsigned char code dispcode[]={0x3f,0x06,0x5b,0x4f,0x66,
0x6d,0x7d,0x07,0x7f,0x6f,0x00,0x40}
unsigned char dispbuf[8]={0,0,0,0,0,0,10,10}
unsigned char temp[8]
unsigned char dispcount
unsigned char T0count
unsigned char timecount
bit flag
unsigned long x
void main(void)
{
unsigned char i
TMOD=0x15
TH0=0
TL0=0
TH1=(65536-4000)/256
TL1=(65536-4000)%256
TR1=1
TR0=1
ET0=1
ET1=1
EA=1
while(1)
{
if(flag==1)
{
flag=0
x=T0count*65536+TH0*256+TL0
for(i=0i<8i++)
{
temp[i]=0
}
i=0
while(x/10)
{
temp[i]=x%10
x=x/10
i++
}
temp[i]=x
for(i=0i<6i++)
{
dispbuf[i]=temp[i]
}
timecount=0
T0count=0
TH0=0
TL0=0
TR0=1
}
}
}
void t0(void) interrupt 1 using 0
{
T0count++
}
void t1(void) interrupt 3 using 0
{
TH1=(65536-4000)/256
TL1=(65536-4000)%256
timecount++
if(timecount==250)
{
TR0=0
timecount=0
flag=1
}
P0=dispcode[dispbuf[dispcount]]
P2=dispbit[dispcount]
dispcount++
if(dispcount==8)
{
dispcount=0
}
}
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)