定时器溢出好像和cy没有关系吧。我记得cy是程序状态字寄存器(psw),是执行算术运算和逻辑运算时判断第7位是否进位的一个标志位。所以cy不会变1
定时器控制寄存器是tcon,tf0属于tcon,用于计数溢出的标志位。
延时程序(包括asm和C程序,都是我在学单片机的过程中用到的),在单片机延时程序中应考虑所使用的晶振的频率,在51系列的单片机中我们常用的是110592MHz和120000MHz的晶振,而在AVR单片机上常用的有8000MHz和4000MH的晶振所以在网上查找程序时如果涉及到精确延时则应该注意晶振的频率是多大。
软件延时:(asm)
晶振12MHZ,延时1秒
程序如下:
DELAY:MOV 72H,#100
LOOP3:MOV 71H,#100
LOOP1:MOV 70H,#47
LOOP0:DJNZ 70H,LOOP0
NOP
DJNZ 71H,LOOP1
MOV 70H,#46
LOOP2:DJNZ 70H,LOOP2
NOP
DJNZ 72H,LOOP3
MOV 70H,#48
LOOP4:DJNZ 70H,LOOP4
定时器延时:
晶振12MHZ,延时1s,定时器0工作方式为方式1
DELAY1:MOV R7,#0AH ;;晶振12MHZ,延时05秒
AJMP DELAY
DELAY2:MOV R7,#14H ;;晶振12MHZ,延时1秒
DELAY:CLR EX0
MOV TMOD,#01H ;设置定时器的工作方式为方式1
MOV TL0,#0B0H ;给定时器设置计数初始值
MOV TH0,#3CH
SETB TR0 ;开启定时器
HERE:JBC TF0,NEXT1
SJMP HERE
NEXT1:MOV TL0,#0B0H
MOV TH0,#3CH
DJNZ R7,HERE
CLR TR0 ;定时器要软件清零
SETB EX0
RET
C语言延时程序:
10ms延时子程序(12MHZ)
void delay10ms(void)
{
unsigned char i,j,k;
for(i=5;i>0;i--)
for(j=4;j>0;j--)
for(k=248;k>0;k--);
}
1s延时子程序(12MHZ)
void delay1s(void)
{
unsigned char h,i,j,k;
for(h=5;h>0;h--)
for(i=4;i>0;i--)
for(j=116;j>0;j--)
for(k=214;k>0;k--);
}
200ms延时子程序(12MHZ)
void delay200ms(void)
{
unsigned char i,j,k;
for(i=5;i>0;i--)
for(j=132;j>0;j--)
for(k=150;k>0;k--);
}
500ms延时子程序程序: (12MHZ)
void delay500ms(void)
{
unsigned char i,j,k;
for(i=15;i>0;i--)
for(j=202;j>0;j--)
for(k=81;k>0;k--);
}
下面是用了80000MHZ的晶振的几个延时程序(用定时0的工作模式1):
(1)延时09MS
void delay_0_9ms(void)
{
TMOD=0x01; /定时器0工作在模式1下(16位计数器)/
TH0=0xfd;
TL0=0xa8;
TR0=1; /启动定时器/
while(TF0==0);
TR0=0;
}
(2)延时1MS
void delay_1ms(void)
{
TMOD=0x01; /定时器0工作在模式1下(16位计数器)/
TH0=0xfd;
TL0=0x65;
TR0=1; /启动定时器/
while(TF0==0);
TR0=0;
}
(3)延时45ms
void delay_4_5ms(void)
{
TMOD=0x01; /定时器0工作在模式1下(16位计数器)/
TH0=0xf4;
TL0=0x48;
TR0=1; /启动定时器/
while(TF0==0);
TR0=0;
}
如果是用汇编语言的话,是可以做到精准延时的。当然首先要有一个精准的时钟,也即是晶振要准确,比如使用精准的12M晶振12T模式的51单片机延时1毫秒,程序如下:
DELAY1MS:
MOV R3,#249 ;1
DJNZ R3,$ ;2249
MOV R3,#249 ;1
DJNZ R3,$ ;2249
RET ;2
延时时间为:1+2249+1+2249+2=1000T,这里使用12M晶振,T=1微秒,所以总共延时1000微秒也即是1毫秒。如果要延时更多时间,可以采用循环方式来实现。当然在计算延时时间时要考虑到使用的单片机类型和型号,比如STC89C51是6T,DS80320是4T,STC12C5A60S2是12T,这些因数一定要考虑进去才行。
计时1小时差小于1秒,也就是误差要求小于1/3600普通的晶振很难有这个精度,使用外部时钟比较有把据。你要的是软件就给你一个吧,能不能达到精度要求就看单片机的时钟部分,跟软件没有关系。
//
/广州一丁自动化设备有限公司020-28969088 /
//
#include <reg52h>
sbit led=P1^0;// led灯
unsigned int num=0;
unsigned int num1=0;
void time0() interrupt 1
{
TH0=(65536-50000)/256; //12M/6=2M,50000/2M=0025秒(6M把50000改为25000)
TL0=(65536-50000)%256;
num++;//0025秒加1
}
void main()
{
TMOD=0x01;//定时器0工作于方式1
TH0=(65536-50000)/256;//赋初值
TL0=(65536-50000)%256;
EA=1;//开总中断
ET0=1;//开定时器0
TR0=1;//运行定时器0
while(1)
{
if(num==40)//40个0025秒等于1秒
{
num=0;
num1++;
if(num1==3600)//24M改7200
{
num1=0;
led=~led;//P10LED灯亮1小时灭1小时
}
}
}
如果是精确的话,需要用定时器,不是太精确的话,软件延时就可以。
假如12M晶振
void delay10s()
{
int i,j;
for(i=0;i<10000;i++)
for(j=0;j<120;j++);
}
void delay10s()
{
unsigned char i;
TMOD=0x01;
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
TR0=1;
i=200;
while(i--)
{
while(TF0==0);
TF0=0;
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
}
}
以上就是关于单片机定时器做精确延时问题全部的内容,包括:单片机定时器做精确延时问题、怎么用C语言写可以调节的延时程序,请大侠帮忙、单片机精准延时汇编语言等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)