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的误差,精度大幅度提高,可以满足大多数的需要! |
上一篇:单片机系统硬件抗干扰实践
下一篇:STM8L外部中断为何死循环 寄存器操作
推荐阅读最新更新时间:2024-03-16 15:19