標題: 單片機匯編語言使用共陽極靜態數碼管按鍵顯示0到9(中斷子程序) [打印本頁]
作者: 布魯克林章魚哥 時間: 2024-11-7 21:50
標題: 單片機匯編語言使用共陽極靜態數碼管按鍵顯示0到9(中斷子程序)
之前寫了一個沒有中斷子程序的版本,想要用中斷子程序但是不清楚該怎么加,求教各位大佬TAT
按鍵顯示1-9(簡易版)
ORG 0000H ; 設置程序的起始地址為0000H
LJMP START ; 無條件跳轉到標號START,開始執行主程序
ORG 0050H ; 程序存儲地址從0050H開始
START:
MOV A,#0FFH ; 將累加器A清零,用于初始化P0口
MOV P0,A ; 將累加器A的值(0FFH)輸出到P0口,熄滅數碼管上的所有段
MOV DPTR,#NUMTAB ; 將數據指針DPTR指向數碼管段碼表NUMTAB的起始地址
MOV R0,#00H ; 初始化R0寄存器,用于存儲當前要顯示的數字
MOV P1,#0FFH ; 將P1口的所有位設置為高電平,準備讀取按鍵狀態
LOPDSP: ; 主循環標簽
MOV A, R0 ; 將當前要顯示的數字(存儲在R0中)加載到累加器A
MOV DPTR,#NUMTAB ; 再次將數據指針DPTR指向數碼管段碼表NUMTAB的起始地址
MOVC A,@A+DPTR ; 根據累加器A的值(當前數字)和DPTR的地址,讀取對應的數碼管編碼到累加器A
MOV P0, A ; 將累加器A中的數碼管編碼輸出到P0口,顯示當前數字
CHECK_BUTTON: ; 檢查按鍵標簽
JB P1.0, KEY_PRESSED ; 檢查P1.0是否被按下(假設低電平有效)
SJMP LOPDSP ; 如果沒有按鍵按下,跳回LOPDSP繼續循環顯示當前數字
KEY_PRESSED: ; 按鍵按下處理標簽
LCALL DELAY ; 調用延時子程序,消除抖動
JB P1.0, CHECK_BUTTON ; 再次檢查按鍵是否仍然被按下
INC R0 ; 數字加1
CJNE R0, #10, SKIP_RESET ; 如果計數器R0達到10,跳轉到SKIP_RESET重置為0
MOV R0, #00H ; 重置計數器R0為0
SKIP_RESET:
SJMP LOPDSP ; 跳回LOPDSP繼續循環
HERE: ; 死循環標簽,用于程序結束后停止
SJMP HERE ; 無限循環
DELAY: ; 延時子程序標簽
MOV R1,#02H ; 設置延時計數器R1為2,用于控制延時長度
LP1: MOV R2,#0FFH ; 設置內部延時計數器R2為255
LP2: MOV R3,#0FFH ; 設置最內層延時計數器R3為255
LP3: DJNZ R3,LP3 ; 遞減R3,直到為0,然后跳回LP3繼續循環
DJNZ R2,LP2 ; 遞減R2,直到為0,然后跳回LP2繼續循環
DJNZ R1,LP1 ; 遞減R1,直到為0,然后跳回LP1繼續循環
RET ; 返回主程序
NUMTAB: ; 數碼管段碼表標簽
DB 0C0H, 0F9H, 0A4H, 0B0H, 99H, 92H,82H,0F8H, 80H, 90H ; 共陽極數碼管0-9的編碼表
END ; 程序結束
使用中斷(寫了一部分但是不知道該怎么繼續了
ORG 0000H ; 設置程序的起始地址為0000H
LJMP START
ORG 0003H
LJMP EXINT0
ORG 0050H
MAIN: MOV SP,#60
SETB IT1
CLR IE1
SETB EX1
SETB EA
HERE: SJMP HERE
LEDTAB: DB 00H
DELAY: ; 延時子程序標簽
MOV R1,#02H; 設置延時計數器R1為2,用于控制延時長度
LP1: MOVR2,#0FFH ; 設置內部延時計數器R2為255
LP2: MOVR3,#0FFH ; 設置最內層延時計數器R3為255
LP3: DJNZR3,LP3 ; 遞減R3,直到為0,然后跳回LP3繼續循環
DJNZ R2,LP2; 遞減R2,直到為0,然后跳回LP2繼續循環
DJNZ R1,LP1; 遞減R1,直到為0,然后跳回LP1繼續循環
RET ; 返回主程序
EXINT0: PUSH PSW
LCALL DELAY
JB P3.3,EXIT
中斷子程序主體
EXIT: POP PSW
RET1
NUMTAB: ; 數碼管段碼表標簽
DB 0C0H, 0F9H, 0A4H, 0B0H, 99H, 92H,82H,0F8H, 80H, 90H ; 共陽極數碼管0-9的編碼表
END ; 程序結束
作者: 飛云居士 時間: 2024-11-8 10:41
用中斷來實現數碼管顯示和按鍵檢測的功能,需要進行下面的修改。以下是使用外部中斷0(INT0)來實現按鍵檢測和數碼管顯示的8051匯編代碼示例:
```assembly
ORG 0000H ; 設置程序的起始地址為0000H
LJMP START ; 無條件跳轉到標號START,開始執行主程序
ORG 0003H ; 外部中斷0的中斷向量地址
LJMP EXINT0 ; 跳轉到外部中斷0的處理程序
ORG 0050H ; 程序存儲地址從0050H開始
START:
MOV SP, #60H ; 初始化堆棧指針
MOV A, #0FFH ; 將累加器A清零,用于初始化P0口
MOV P0, A ; 將累加器A的值(0FFH)輸出到P0口,熄滅數碼管上的所有段
MOV DPTR, #NUMTAB ; 將數據指針DPTR指向數碼管段碼表NUMTAB的起始地址
MOV R0, #00H ; 初始化R0寄存器,用于存儲當前要顯示的數字
MOV P1, #0FFH ; 將P1口的所有位設置為高電平,準備讀取按鍵狀態
SETB IT0 ; 設置INT0為邊沿觸發模式
CLR EX0 ; 清除外部中斷0的請求標志
SETB EA ; 開啟全局中斷
SETB EX0 ; 開啟外部中斷0
MAIN_LOOP:
SJMP MAIN_LOOP ; 主循環,等待中斷
; 外部中斷0的處理程序
EXINT0:
PUSH PSW ; 保存程序狀態字
PUSH ACC ; 保存累加器
PUSH B ; 保存B寄存器
PUSH DPL ; 保存數據指針低字節
PUSH DPH ; 保存數據指針高字節
LCALL DELAY ; 調用延時子程序,消除抖動
JB P1.0, EXIT_INT ; 檢查P1.0是否被按下(假設低電平有效)
INC R0 ; 數字加1
MOV A, R0
CJNE A, #10, DISPLAY ; 如果計數器R0達到10,跳轉到DISPLAY重置為0
MOV R0, #00H ; 重置計數器R0為0
DISPLAY:
MOV A, R0
MOVC A, @A+DPTR ; 根據累加器A的值(當前數字)和DPTR的地址,讀取對應的數碼管編碼到累加器A
MOV P0, A ; 將累加器A中的數碼管編碼輸出到P0口,顯示當前數字
EXIT_INT:
POP DPH ; 恢復數據指針高字節
POP DPL ; 恢復數據指針低字節
POP B ; 恢復B寄存器
POP ACC ; 恢復累加器
POP PSW ; 恢復程序狀態字
CLR EX0 ; 清除外部中斷0的請求標志
RETI ; 從中斷返回
DELAY: ; 延時子程序標簽
MOV R1, #02H ; 設置延時計數器R1為2,用于控制延時長度
LP1: MOV R2, #0FFH ; 設置內部延時計數器R2為255
LP2: MOV R3, #0FFH ; 設置最內層延時計數器R3為255
LP3: DJNZ R3, LP3 ; 遞減R3,直到為0,然后跳回LP3繼續循環
DJNZ R2, LP2 ; 遞減R2,直到為0,然后跳回LP2繼續循環
DJNZ R1, LP1 ; 遞減R1,直到為0,然后跳回LP1繼續循環
RET ; 返回主程序
NUMTAB: ; 數碼管段碼表標簽
DB 0C0H, 0F9H, 0A4H, 0B0H, 99H, 92H, 82H, 0F8H, 80H, 90H ; 共陽極數碼管0-9的編碼表
END ; 程序結束
```
代碼中設置了外部中斷0(INT0)來檢測按鍵的按下。當INT0被觸發時,程序會跳轉到`EXINT0`處理程序。在這個處理程序中,我們首先執行延時以消除按鍵抖動,然后檢查按鍵是否仍然被按下。如果按鍵被按下,程序會增加計數器`R0`,并根據`R0`的值顯示相應的數碼管編碼。如果`R0`的值達到10,它會被重置為0。最后,我們清除中斷請求標志并從中斷返回。
| 歡迎光臨 (http://www.denmoz.com/bbs/) |
Powered by Discuz! X3.1 |