跪求单片机at89c52频率计c语言程序

跪求单片机at89c52频率计c语言程序,第1张

方波叫函数发生器。频率计程序如下:

#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

}

}


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/yw/7746974.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-04-09
下一篇2023-04-09

发表评论

登录后才能评论

评论列表(0条)

    保存