#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++
}
定时器1对外部脉冲计数时TMOD高4位设置应该是5因此TMOD=0x51
以下我的频率计程序:
#include <reg52.h>//因没用到STC12C5410专有特殊功能寄存器,此处用52或51的头文件均可
#define unit unsigned int
#define uchar unsigned char
//定义以I/O口的功能
sbit beiguang=P3^2//液晶屏背光
sbit rs=P1^3//液晶屏写选择,0命令 1数据
sbit rw=P1^4//液晶屏读写选择
sbit lcden=P1^5//液晶屏使能
sbit fm=P1^7//蜂鸣器
#define db P2 //定义P2为数据输出口,写数据时用db代替P2,增加液晶屏程序的通用性
//更改硬件接线时,只更改此处,而不必去更改液晶屏读写子程序
uchar aa,bb,cc//变量声明
unit dd,ee
void Delay1ms(unsigned int i) //1ms延时程序
{
unsigned int j
for(i>0i--)
{
for(j=0j<125j++)
{}
}
}
void init()//初始化设置
{
TMOD=0x15//定时器0作为计数器,定时器1作为定时器用
TH0=0//计数器清0
TL0=0
EA=1//开总中断
ET1=1//允许定时器1中断
TH1=0x4c
TL1=0x5c
TR0=1//启动计数器
TR1=1//启动定时器
aa=0
}
void write_com(uchar com)//向液晶屏写命令
{
db=com
rs=0
rw = 0
lcden=0
Delay1ms(10*12)
lcden=1
Delay1ms(10*12)
lcden=0
}
void write_date(uchar date)//向液晶屏写数据
{
db=date
rs=1
rw = 0
lcden=0
Delay1ms(10*12)
lcden=1
Delay1ms(10*12)
lcden=0
}
void init2()//液晶屏初始化
{
beiguang=0
rw=0
write_com(0x38)
Delay1ms(10*12)
write_com(0x0f)
Delay1ms(10*12)
write_com(0x06)
Delay1ms(10*12)
write_com(0x01)
Delay1ms(10*12)
}
void display4(unsigned int number) //单行多位显示程序
{
uchar A1,A2,A3,A4,A5
init2()//液晶屏初始化
A1=number/10000%10//分离出万,千,百,十,个,对于int型数据,最大不超过65535
A2=number/1000%10
A3=number/100%10
A4=number/10%10
A5=number%10
write_com(0x80)//第1个数据的位置设定,第1行第1列
Delay1ms(10)
write_date(0x30+A1)//写数据
Delay1ms(10)
write_date(0x30+A2)
Delay1ms(10)
write_date(0x30+A3)
Delay1ms(10)
write_date(0x30+A4)
Delay1ms(10)
write_date(0x30+A5)
Delay1ms(10)
write_com(0x87)//第6个数据'H'的位置,中间空85和86 二格
write_date('H')
Delay1ms(10)
write_date('z')
Delay1ms(10)
}
void main()//主程序很简单
{
init()//初始化
while(1)//循环程序
{
dd=bb*256+cc//0.5S的计数值
ee=2*dd//换算为1秒钟的计数值
if(aa==1)
{
if(TH0>12)//预判断,50ms内TH0>12,1s内计数值将超过可计数的最大值65535
fm=0//报警
}
display4(ee)//显示
fm=1//报警停止
}
}
void timer1()interrupt 3//注意:定时器1的中断序号为3
{
aa++
TH1=0x4c//11.0592Mhz
TL1=0x5c
if(aa==10)//中断10次,共0.5S
{
TR0=0//暂停计数
aa=0
bb=TH0//读出计数器数据
cc=TL0
TL0=0//计数器清0
TH0=0
TR0=1//重新启动
}
}
你应该是T0做,T1另外有用吧?我花了一个小时,调了一个,只计算了频率,频率低是1秒计算一次,你可以改,部分地方应该可以优化。你试试,如果可以可以把keil的项目打包发给你。程序如下:
/*
testT0T1.c
芯片:AT89C51
晶振:12MHz
*/
#include <reg51.h>
bit g_bitNewF//计算了新频率
unsigned int g_Count
unsigned int g_Timer200
unsigned long frequency
void Init(void)
{
EA = 0
//T0使用模式3,TL0计数,
TMOD = 0x07
TL0 = 0x00
//TH0计时200us
TH0 = 0x38
//启动
TF0 = 0
TF1 = 0//可以不清这两个
TR0 = 1
TR1 = 1
EA = 1
ET0 = 1
ET1 = 1
}
void main (void)
{
g_bitNewF = 0
g_Timer200 = 0
Init()
while(1)
{
if (g_bitNewF)
{
g_bitNewF = 0
//display or do something else.
}
}
}
//计数满中断
void T0_Int(void) interrupt 1
{
TL0 = 0x00
g_Count += 255
if (g_Count >65535-255) //频率大提前计算
{
g_bitNewF = 1
if (g_Timer200<10){return}//误差太大,请自己增加出错处理
frequency = ((long)g_Count*5000)/g_Timer200
}
}
//定时到中断
void T1_Int(void) interrupt 3
{
TH0 = 0x46
g_Timer200++
if (g_Timer200 == 5000)
{
g_Timer200 = 0
g_bitNewF = 1
frequency = g_Count + TL0
TL0 = 0
g_Count = 0
}
}
//end
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)