1、查看一下你的硬件环境,比如晶振大小,CPU型号,不用CPU指令的机器周期是不一样的。
2、计算延时需要的机器周期。比如采用12M晶振,CPU采用通用8051,那么一个机器周期为1US,50毫秒为50*1000=50000US,需要的机器周期=50000/1=50000。
3、试编程,如下:
程序代码指令时间 总共时间
DELAY50MS: 2 2
MOV R7,#A 1 1
DELAY1:
MOV R6,#B 1 1*A
DJNZ R6,$ 2 2*B*A
DJNZ R7,DELAY1 2 2*A
RET 2 2
所以总时间=2+1+A+2*A*B+2*A+2=5+3A+2AB
4、凑数求A、B
根据2、3得到如下式子:
50000=5+3A+2AB
可以有很多种结果,不过最好是以A尽可能小,B尽可能大为原则,当然不能大于255.
我现在凑出A=110,B=225;那么总延时时间=5+3*110+2*110*225=49835。还差165US
5、补齐不够时间
再加一个小循环就OK了,呵呵如下:
MOV R6,#C
DJNZ R6,$
会算了吧,2*C+1=165;所以C=82。
现在完整的延时程序出来了,如下:
DELAY50MS: 2 2
MOV R7,#110 1 1
DELAY1:
MOV R6,#225 1 1*110
DJNZ R6,$ 2 2*225*110
DJNZ R7,DELAY1 2 2*110
MOV R6,#821 1
DJNZ R6,$22*82
RET 2 2
很圆满:总的时间50000微妙,也就是50毫秒。这种方式编程,在该硬件环境下可以保证最大误差为1微妙。
延时程序(包括asm和C程序,都是我在学单片机的过程中用到的),在单片机延时程序中应考虑所使用的晶振的频率,在51系列的单片机中我们常用的是11.0592MHz和12.0000MHz的晶振,而在AVR单片机上常用的有8.000MHz和4.000MH的晶振所以在网上查找程序时如果涉及到精确延时则应该注意晶振的频率是多大。软件延时:(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,延时0.5秒
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=5i>0i--)
for(j=4j>0j--)
for(k=248k>0k--)
}
1s延时子程序(12MHZ)
void delay1s(void)
{
unsigned char h,i,j,k
for(h=5h>0h--)
for(i=4i>0i--)
for(j=116j>0j--)
for(k=214k>0k--)
}
200ms延时子程序(12MHZ)
void delay200ms(void)
{
unsigned char i,j,k
for(i=5i>0i--)
for(j=132j>0j--)
for(k=150k>0k--)
}
500ms延时子程序程序: (12MHZ)
void delay500ms(void)
{
unsigned char i,j,k
for(i=15i>0i--)
for(j=202j>0j--)
for(k=81k>0k--)
}
下面是用了8.0000MHZ的晶振的几个延时程序(用定时0的工作模式1):
(1)延时0.9MS
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)延时4.5ms
void delay_4_5ms(void)
{
TMOD=0x01/*定时器0工作在模式1下(16位计数器)*/
TH0=0xf4
TL0=0x48
TR0=1/*启动定时器*/
while(TF0==0)
TR0=0
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)