×
嵌入式 > 技术百科 > 详情

C51中延时程序的编写

发布时间:2020-06-16 发布时间:
|

C51及C51和汇编的混合编程的资料都是从前辈那儿偷来的,先谢谢各位前辈的分享!:)

 

 
 
 
 众所周知,c51在编程时很难掌握程序运行的时间,所以编写延时程序时很难做到很精确,解决的方式有插入汇编语句,但总觉得汇编用起来不是那么顺手,还可以用定时器来做,但只是一个很小的延时程序而已,不必小题大作。其实办法还是有的,下面介绍一下常用的几种延时程序。
1、us级延时程序
   常用的一个函数如下:
   void delayus(unsigned char x)
   {  while(--x) ;
   }
生成的汇编代码为:
C:0x001C    7F0A        MOV      R7,#0x0A                 //2us
C:0x001E    12003E     LCALL    delayus(C:003E)    //   2us
C:0x003E    DFFE        DJNZ     R7,delayus(C:003E)   //2x us
C:0x0040    22              RET                                        // 1us
      所以调用一次函数延时时间为 ( 2x+5 )us ,可以用来延时大于5us的时间。
注意的是x要是unsigned char 类型,且 --x 不能写成x-- ,否则汇编代码会有一大串:C:0x001C    7F02      MOV      R7,#0x02
C:0x001E    120032   LCALL    delayus(C:0032)
C:0x0032    AE07       MOV      R6,0x07
C:0x0034    1F           DEC      R7
C:0x0035    EE           MOV      A,R6
C:0x0036    70FA       JNZ      delayus(C:0032)
C:0x0038    22            RET      
       因为汇编中的DJNZ 语句是先减一再判断的,和-xx的算法一致,所以--x和x--相差了很多。上述函数只适合x的范围是0-255,如果需要延时大于255*2+5us,则可以连续调用几次函数。
2、ms的延时程序
       常用的一个函数:
void delayms(unsigned int x)
{
unsigned char i;
  while(x--)
    {
      for(i=0;i<125;i++){;}
    }
}
      我们来看一下它的精度如何
x                     us
1                 1024
5                 5076
10             10141
50             50661
100         101311
      可见,精度不“精”,随着x值的增大,延时误差越大,只适合在延时不要求很准确的地方使用。于是想到对程序进行一些修改,想到125是不是取的太大了,假如把它变为可变,不同的x值取值不同,就可以对延时时间进行一些修正,把程序改为如下:
   void delayms(unsigned int x,unsigned char y)
{
unsigned char j;
 while(x--)
    { 
      for(j=0;j     }
}
      对不同的延时时间取不同的y值,发现一个“怪”现象,当取y=123时,延时时间的误差是一个固定值:
x                     us
1                 1017
2                 2016
5                 5016
10             10016
50             50016
100         100016
    除了x=1外,其余的误差都为16us, 即不管x为多大都只有16us的误差,精度大幅度提高,可以满足大多数的需要!

 

『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
FPGA及CPLD应用领域不断拓展