51單片機(jī)定時、中斷練習(xí):用延時程序做閃爍燈
上面的兩個程序運(yùn)行后,我們發(fā)現(xiàn)燈的閃爍非常快,根本分辨不出來,只是視覺上感到燈有些晃動而已,為什么呢?我們可以計算一下,定時器中預(yù)置的數(shù)是5536,所以每計60000個脈沖就是定時時間到,這60000個脈沖的時間是多少呢?我們的晶振是12M,所以就是60000微秒,即60毫秒,因此速度是非常快的。如果我想實現(xiàn)一個1S的定時,該怎么辦呢?在該晶振瀕率下,最長的定時也就是65。536個毫秒啊!上面給出一個例子。
ORG0000H
AJMPSTART
ORG000BH;定時器0的中斷向量地址
AJMPTIME0;跳轉(zhuǎn)到真正的定時器程序處
ORG30H
START:
MOVP1,#0FFH;關(guān)所燈
MOV30H,#00H;軟件計數(shù)器預(yù)清0
MOVTMOD,#00000001B;定時/計數(shù)器0工作于方式1
MOVTH0,#3CH
MOVTL0,#0B0H;即數(shù)15536
SETBEA;開總中斷允許
SETBET0;開定時/計數(shù)器0允許
SETBTR0;定時/計數(shù)器0開始運(yùn)行
LOOP:AJMPLOOP;真正工作時,這里可寫任意程序
TIME0:;定時器0的中斷處理程序
PUSHACC
PUSHPSW;將PSW和ACC推入堆棧保護(hù)
INC30H
MOVA,30H
CJNEA,#20,T_RET;30H單元中的值到了20了嗎?
T_L1:CPLP1.0;到了,取反P10
MOV30H,#0;清軟件計數(shù)器
T_RET:
MOVTH0,#15H
MOVTL0,#9FH;重置定時常數(shù)
POPPSW
POPACC
RETI
END
先自己分析一下,看看是怎么實現(xiàn)的?這里采用了軟件計數(shù)器的概念,思路是這樣的,先用定時/計數(shù)器0做一個50毫秒的定時器,定時是間到了以后并不是立即取反P10,而是將軟件計數(shù)器中的值加1,如果軟件計數(shù)器計到了20,就取反P10,并清掉軟件計數(shù)器中的值,否則直接返回,這樣,就變成了20次定時中斷才取反一次P10,因此定時時間就延長了成了20*50即1000毫秒了。
這個思路在工程中是非常有用的,有的時候我們需要若干個定時器,可51中總共才有2個,怎么辦呢?其實,只要這幾個定時的時間有一定的公約數(shù),我們就可以用軟件定時器加以實現(xiàn),如我要實現(xiàn)P10口所接燈按1S每次,而P11口所接燈按2S每次閃爍,怎么實現(xiàn)呢?對了我們用兩個計數(shù)器,一個在它計到20時,取反P10,并清零,就如上面所示,另一個計到40取反P11,然后清0,不就行了嗎?這部份的程序如下
ORG0000H
AJMPSTART
ORG000BH;定時器0的中斷向量地址
AJMPTIME0;跳轉(zhuǎn)到真正的定時器程序處
ORG30H
START:
MOVP1,#0FFH;關(guān)所燈
MOV30H,#00H;軟件計數(shù)器預(yù)清0
MOVTMOD,#00000001B;定時/計數(shù)器0工作于方式1
MOVTH0,#3CH
MOVTL0,#0B0H;即數(shù)15536
SETBEA;開總中斷允許
SETBET0;開定時/計數(shù)器0允許
SETBTR0;定時/計數(shù)器0開始運(yùn)行
LOOP:AJMPLOOP;真正工作時,這里可寫任意程序
TIME0:;定時器0的中斷處理程序
PUSHACC
PUSHPSW;將PSW和ACC推入堆棧保護(hù)
INC30H
INC31H;兩個計數(shù)器都加1
MOVA,30H
CJNEA,#20,T_NEXT;30H單元中的值到了20了嗎?
T_L1:CPLP1.0;到了,取反P10
MOV30H,#0;清軟件計數(shù)器
T_NEXT:
MOVA,31H
CJNEA,#40,T_RET;31h單元中的值到40了嗎?
T_L2:
CPLP1.1
MOV31H,#0;到了,取反P11,清計數(shù)器,返回
T_RET:
MOVTH0,#15H
MOVTL0,#9FH;重置定時常數(shù)
POPPSW
POPACC
RETI
END
塵埃粒子計數(shù)器相關(guān)文章:塵埃粒子計數(shù)器原理
評論