您的位置 首页 基础

如何用定时器来完成灯的闪耀功用

如何用定时器来实现灯的闪烁功能-键入程序,看到了什么?灯在闪烁了,这可是用定时器做的,不再是主程序的循环了。简单地分析一下程序,为什么用JBC呢?TF0是定时/计数器0的溢出标记位,当定时器产生溢出后,该位由0变1,所以查询该位就可知宇时时间是否已到。该位为1后,要用软件将标记位清0,以便下一次定时是间到时该位由0变1,所以用了JBC指令,该指位在判1转移的同时,还将该位清0。

本站前面有个灯的闪耀编程实例,那是用延时程序做的,现在回想起来,这样做不很恰当,为什么呢?咱们的主程序做了灯的闪耀,就不能再干其它的事了,莫非单片机只能这样作业吗?当然不是,咱们能用守时器来完成灯的闪耀的功用。

如何用守时器来完成灯的闪耀功用
 

例1:查询方法

ORG 0000H

AJMP START

ORG 30H

START:

MOV P1,#0FFH ;关所 灯

MOV TMOD,#00000001B ;守时/计数器0作业于方法1

MOV TH0,#15H

MOV TL0,#0A0H ;即数5536

SETB TR0 ;守时/计数器0开端运转

LOOP:JBC TF0,NEXT ;假如TF0等于1,则清TF0并转NEXT处

AJMP LOOP ;否则跳转到LOOP处运转

NEXT:CPL P1.0

MOV TH0,#15H

MOV TL0,#9FH;重置守时/计数器的初值

AJMP LOOP

END AJMP LOOP

END

键入程序,看到了什么?灯在闪耀了,这可是用守时器做的,不再是主程序的循环了。简略地剖析一下程序,为什么用JBC呢?TF0是守时/计数器0的溢出符号位,当守时器发生溢出后,该位由0变1,所以查询该位就可知宇时时刻是否已到。该位为1后,要用软件将符号位清0,以便下一次守时是间届时该位由0变1,所以用了JBC指令,该指位在判1搬运的一起,还将该位清0。

以上程序是能完成灯的闪耀了,可是主程序除了让灯闪耀外,仍是不能做其他的事啊!不,不对,咱们能在LOOP:……和AJMP LOOP指令之间刺进一些指令来做其他的作业,只需确保履行这些指令的时刻少于守时时刻就行了。那咱们在用软件延时程序的时分不是也能用一些指令来代替DJNZ吗?是的,可是那就要求你准确核算所用指令的时刻,然后再减去对应的DJNZ循环次数,很不便利,而现在只需求所用指令的时刻少于守时时刻就行,明显要求低了。当然,这样的方法仍是欠好,所以咱们常用以下的方法来完成。

程序2:用中止完成

ORG 0000H ,

AJMP START

ORG 000BH ;守时器0的中止向量地址

AJMP TIME0 ;跳转到真实的守时器程序处

ORG 30H

START:

MOV P1,#0FFH ;关所 灯

MOV TMOD,#00000001B ;守时/计数器0作业于方法1

MOV TH0,#15H

MOV TL0,#0A0H ;即数5536

SETB EA ;开总中止答应

SETB ET0 ;开守时/计数器0答应

SETB TR0 ;守时/计数器0开端运转

LOOP: AJMP LOOP ;真实作业时,这儿可写恣意程序

TIME0: ;守时器0的中止处理程序

PUSH ACC

PUSH PSW ;将PSW和ACC推入仓库维护

CPL P1.0

MOV TH0,#15H

MOV TL0,#0A0H ;重置守时常数

POP PSW

POP ACC

RETI

END

上面的例程中,守时时刻一到,TF0由0变1,就会引发中止,CPU将主动转至000B处寻觅程序并履行,因为留给守时器中止的空间只需8个字节,明显不足以写下一切有中止处理程序,所以在000B处组织一条跳转指令,转到实践处理中止的程序处,这样,中止程序能写在恣意当地,也能写恣意长度了。进入守时中止后,首先要保存当时的一些状况,程序中只演示了保存存ACC和PSW,实践作业中应该根据需求将可能会改动的单元的值都推入仓库进行维护(本程序中实践不需保存护任何值,这儿只作个演示)。

上面的两个单片机程序运转后,咱们发现灯的闪耀非常快,底子分辩不出来,仅仅视觉上感到灯有些晃动罢了,为什么呢?咱们能核算一下,守时器中预置的数是5536,所以每计60000个脉冲便是守时时刻到,这60000个脉冲的时刻是多少呢?咱们的晶体震动器是12M,所以便是60000微秒,即60毫秒,因而速度对错常快的。假如我想完成一个1S的守时,该怎样办呢?在该晶体震动器濒率下,最长的守时也便是65。536个毫秒啊!上面给出一个例程。

ORG 0000H

AJMP START

ORG 000BH ;守时器0的中止向量地址

AJMP TIME0 ;跳转到真实的守时器程序处

ORG 30H

START:

MOV P1,#0FFH ;关所 灯

MOV 30H,#00H ;软件计数器预清0

MOV TMOD,#00000001B ;守时/计数器0作业于方法1

MOV TH0,#3CH

MOV TL0,#0B0H ;即数15536

SETB EA ;开总中止答应

SETB ET0 ;开守时/计数器0答应

SETB TR0 ;守时/计数器0开端运转

LOOP: AJMP LOOP ;真实作业时,这儿可写恣意程序

TIME0: ;守时器0的中止处理程序

PUSH ACC

PUSH PSW ;将PSW和ACC推入仓库维护

INC 30H

MOV A,30H

CJNE A,#20,T_RET ;30H单元中的值到了20了吗?

T_L1: CPL P1.0 ;到了,取反P10

MOV 30H,#0 ;清软件计数器

T_RET:

MOV TH0,#15H

MOV TL0,#9FH ;重置守时常数

POP PSW

POP ACC

RETI

END

先自己剖析一下,看看是怎样完成的?这儿采用了软件计数器的概念,思路是这样的,先用守时/计数器0做一个50毫秒的守时器,守时是间到了今后并不是当即取反P10,而是将软件计数器中的值加1,假如软件计数器计到了20,就取反P10,并清掉软件计数器中的值,否则直接回来,这样,就变成了20次守时中止才取反一次P10,因而守时时刻就延长了成了20*50即1000毫秒了。

这个思路在工程中对错常有用的,有的时分咱们需求若干个守时器,可51中一共才有2个,怎样办呢?其实,只需这几个守时的时刻有必定的公约数,咱们就能用软件守时器加以完成,如我要完成P10口所接灯按1S每次,而P11口所接灯按2S每次闪耀,怎样完成呢?对了咱们用两个计数器,一个在它计到20时,取反P10,并清零,就如上面所示,另一个计到40取反P11,然后清0,不就行了吗?这部份的程序如下

ORG 0000H

AJMP START

ORG 000BH ;守时器0的中止向量地址

AJMP TIME0 ;跳转到真实的守时器程序处

ORG 30H

START:

MOV P1,#0FFH ;关所 灯

MOV 30H,#00H ;软件计数器预清0

MOV TMOD,#00000001B ;守时/计数器0作业于方法1

MOV TH0,#3CH

MOV TL0,#0B0H ;即数15536

SETB EA ;开总中止答应

SETB ET0 ;开守时/计数器0答应

SETB TR0 ;守时/计数器0开端运转

LOOP: AJMP LOOP ;真实作业时,这儿可写恣意程序

TIME0: ;守时器0的中止处理程序

PUSH ACC

PUSH PSW ;将PSW和ACC推入仓库维护

INC 30H

INC 31H ;两个计数器都加1

MOV A,30H

CJNE A,#20,T_NEXT ;30H单元中的值到了20了吗?

T_L1: CPL P1.0 ;到了,取反P10

MOV 30H,#0 ;清软件计数器

T_NEXT:

MOV A,31H

CJNE A,#40,T_RET ;31h单元中的值到40了吗?

T_L2:

CPL P1.1

MOV 31H,#0 ;到了,取反P11,清计数器,回来

T_RET:

MOV TH0,#15H

MOV TL0,#9FH ;重置守时常数

POP PSW

POP ACCm

RETI

END
来历;21ic

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/zhishi/jichu/344026.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部