本文是在上一篇文章基础上的延伸,如果你还不懂单片机控制led的基本原理的话请先看看上一篇文章
https://blog.csdn.net/haohulala/article/details/91372488
有了上一步的基础,我们现在就可以很轻松地开始写滚动显示的程序啦,为了方便起见,我们就写左移显示吧。
part1 如何实现滚动
我们以向左滚动为例,实际上仔细观察后就会发现,想要实现向左滚动就是需要将字形码的起始地址向右偏移,这样汉字左边的部分就会消失,后面的汉字由于偏移就会出现出来。更直观一点说,你可以把屏幕想象成一个窗口,字向左滚动就相当于屏幕向右滚动,也就是起始指针向右偏移。
part2 主程序
;==========
;初始化程序
;==========
MAIN: MOV SP, #5FH ;初始化堆栈指针
MOV IE, #82H ;1000 0010B,表示开总中断和EX1外部中断1
M1: MOV R7, #4H ;****这里控制要显示的字数****
MOV R6, #00H ;这是DPTR的偏移量,这个值不需要修改
M2: ACALL DISPLAY
DJNZ R7, M2
SJMP M1
主程序比step多了一个R6寄存器,用来存放字形码偏移量,赋初值为0
part3 显示程序
;===============================
;显示主程序
;显存为30H开始的32个内存单元
;P0口控制低8位字形码
;P1口控制高8位字形码
;P2口控制字位码
;P2.7口为高电平的时候不选通任何一列
;先送字形码,后送字位码
;然后调用延时子程序后再显示下一列
;显示完一个字后
;================================
DISPLAY:
PUSH 00H
PUSH 01H
PUSH 02H
PUSH 03H
PUSH 04H
PUSH 05H
MOV R5, #10H ;一个字显示16次能显示完,每次都向左偏移一个单位
D3: ACALL GET ;先将下一帧送入显存
MOV R4, #10H ;****这里控制循环次数,也就是一个字显示多久****
;初始化部分
D1: MOV R0, #30H ;用作上半部分显示内容指针
MOV R1, #30H+10H;用作下半部分显示内容指针
MOV R2, #10H ;进行显示内容控制
MOV R3, #00H ;用作字位码
CLR A
MOV P0, A
MOV P1, A
SETB P2.7
;显示部分
D2: MOV P0, @R0 ;将低8位字形码送入P0口
MOV P1, @R1 ;将高8位字形码送入P1口
MOV P2, R3 ;将字位码送入P2口
CLR P2.7 ;将P2.7口置0表示可以显示
ACALL DELAY5 ;延时0.5ms
SETB P2.7 ;关闭显示
INC R0
INC R1
INC R3
DJNZ R2, D2
DJNZ R4, D1
DJNZ R5, D3 ;控制一个字显示16次,每次偏移一个单位
POP 05H
POP 04H
POP 03H
POP 02H
POP 01H
POP 00H
RET
显示程序和step1基本差不多,就是多了一步R5用来控制一个字的显示,每次显示一遍字都会向左移动一格,这样一来循环16次之后,一个字就完全移动出屏幕了,所以我们就用16次来作为一个周期,也就是说一个字一个周期。除此之外,都和step1的代码差不多
part4 送数程序
;================================
;子程序名:送数子程序
;将字形码送入30H开始的32个内存单元
;实现滚动只要是修改这部分代码
;用到R1 R2 A DPTR
;================================
GET: PUSH 01H ;将R1的值入栈
PUSH 02H ;将R2的值入栈
PUSH 03H
;先送上16个字节
MOV R1, #30H ;指向显存的起始地址
MOV R2, #10H ;控制送数个数
MOV DPTR, #TAB1 ;每一帧都比上一帧偏移一个单位
PUSH 06H
I1: INC DPTR ;加上偏移量
DJNZ R6, I1
POP 06H
G1: CLR A
MOVC A, @A+DPTR
MOV @R1, A
INC DPTR
INC R1
DJNZ R2, G1
;接着送下16字节
MOV R2, #10H ;控制送数个数
MOV DPTR, #TAB2
PUSH 06H
I2: INC DPTR ;加上偏移量
DJNZ R6, I2
POP 06H
G2: CLR A
MOVC A, @A+DPTR
MOV @R1, A
INC DPTR
INC R1
DJNZ R2, G2
;完成出栈后返回
POP 03H
POP 02H
POP 01H
INC R6
RET
step1和step2最大的区别就是送数程序了,在step1中只需要简单的将32个数送入显存就行了,step2还需要计算偏移量。
偏移量就是DPTR起始地址距离TAB标签的字节数,没经过一个循环就加一。
但是如果用step1的字形表的话还是会出问题,因为我们直接是加一的,但是字形码的上半部分和下半部分是联合起来作为32个存储单元存储的,这就导致显示完一个字之后会有一个字的上半部分取了上一个字的下半部分;下半部分取了下一个字的上半部分,这样就会出问题。
为了解决这个问题,我们就需要将字形码上半部分和下半部分分开存储,这样只需要一直加一就可以实现滚动了。
part5 字形码表
;=============
;这里储存字形码
;=============
ORG 1000H
;这是上半部分
TAB1: db 08h, 28h, 48h, 88h, 68h, 18h, 00h,0FCh ;"鸡"
db 06h, 15h, 44h, 84h, 7Eh, 04h, 00h, 00h
db 40h, 20h,0F8h, 07h, 40h, 20h, 18h, 0Fh ;"你"
db 08h,0C8h, 08h, 08h, 28h, 18h, 00h, 00h
db 20h, 20h, 20h, 20h, 20h, 20h, 20h,0FFh ;"太"
db 20h, 20h, 20h, 20h, 20h, 30h, 20h, 00h
db 80h, 88h,0A8h,0A8h,0A9h,0AAh,0AEh,0F8h ;"美"
db 0ACh,0AAh,0ABh,0A8h,0ACh, 88h, 80h, 00h
;这是中间需要预留的一部分
DB 00H, 00H, 00H, 00H, 00H, 00H, 00H, 00H
DB 00H, 00H, 00H, 00H, 00H, 00H, 00H, 00H
;这是下半部分
TAB2: db 10h, 08h, 06h, 01h, 02h, 14h, 10h, 13h ;"鸡"
db 12h, 12h, 1Ah, 52h, 82h, 7Fh, 02h, 00h
db 00h, 00h,0FFh, 00h, 00h, 08h, 04h, 43h ;"你"
db 80h, 7Fh, 00h, 01h, 06h, 0Ch, 00h, 00h
db 40h, 40h, 20h, 20h, 10h, 0Ch, 0Bh, 30h ;"太"
db 03h, 0Ch, 10h, 10h, 20h, 60h, 20h, 00h
db 80h, 84h, 84h, 44h, 44h, 24h, 14h, 0Fh ;"美"
db 14h, 24h, 24h, 44h, 46h,0C4h, 40h, 00h
延时程序都是一样的,所以我这里就不介绍了。
part6 完整代码
;==================================
;程序名:51单片机控制16*16LED点阵显示
;作者: 呼啦啦
;完成时间: 2019-6-10
;step 2-left shift
;完成功能:左移显示汉字(最多显示8个字)
;==================================
;============================================
;基本功能介绍:
;使用74HC154进行列选通,p2.0-p2.3作为地址输入口
;一共有16个输出位,分别控制16列
;p0口作为上8行字形码输出口
;p1口作为下8行字形码输出口
;============================================
;==========
;常量表
;常量表报错
;不知道为什么
;==========
;===========
;中断向量表
;===========
ORG 0000H
SJMP MAIN ;跳转到主程序
ORG 0003H ;外中断0
RETI
ORG 000BH ;定时器0
RETI
ORG 0013H ;外中断1
RETI
ORG 001BH ;定时器1
RETI
ORG 0023H ;串行口中断
RETI
;==========
;初始化程序
;==========
MAIN: MOV SP, #5FH ;初始化堆栈指针
MOV IE, #82H ;1000 0010B,表示开总中断和EX1外部中断1
M1: MOV R7, #4H ;****这里控制要显示的字数****
MOV R6, #00H ;这是DPTR的偏移量,这个值不需要修改
M2: ACALL DISPLAY
DJNZ R7, M2
SJMP M1
;===============================
;显示主程序
;显存为30H开始的32个内存单元
;P0口控制低8位字形码
;P1口控制高8位字形码
;P2口控制字位码
;P2.7口为高电平的时候不选通任何一列
;先送字形码,后送字位码
;然后调用延时子程序后再显示下一列
;显示完一个字后
;================================
DISPLAY:
PUSH 00H
PUSH 01H
PUSH 02H
PUSH 03H
PUSH 04H
PUSH 05H
MOV R5, #10H ;一个字显示16次能显示完,每次都向左偏移一个单位
D3: ACALL GET ;先将下一帧送入显存
MOV R4, #10H ;****这里控制循环次数,也就是一个字显示多久****
;初始化部分
D1: MOV R0, #30H ;用作上半部分显示内容指针
MOV R1, #30H+10H;用作下半部分显示内容指针
MOV R2, #10H ;进行显示内容控制
MOV R3, #00H ;用作字位码
CLR A
MOV P0, A
MOV P1, A
SETB P2.7
;显示部分
D2: MOV P0, @R0 ;将低8位字形码送入P0口
MOV P1, @R1 ;将高8位字形码送入P1口
MOV P2, R3 ;将字位码送入P2口
CLR P2.7 ;将P2.7口置0表示可以显示
ACALL DELAY5 ;延时0.5ms
SETB P2.7 ;关闭显示
INC R0
INC R1
INC R3
DJNZ R2, D2
DJNZ R4, D1
DJNZ R5, D3 ;控制一个字显示16次,每次偏移一个单位
POP 05H
POP 04H
POP 03H
POP 02H
POP 01H
POP 00H
RET
;================================
;子程序名:送数子程序
;将字形码送入30H开始的32个内存单元
;实现滚动只要是修改这部分代码
;用到R1 R2 A DPTR
;================================
GET: PUSH 01H ;将R1的值入栈
PUSH 02H ;将R2的值入栈
PUSH 03H
;先送上16个字节
MOV R1, #30H ;指向显存的起始地址
MOV R2, #10H ;控制送数个数
MOV DPTR, #TAB1 ;每一帧都比上一帧偏移一个单位
PUSH 06H
I1: INC DPTR ;加上偏移量
DJNZ R6, I1
POP 06H
G1: CLR A
MOVC A, @A+DPTR
MOV @R1, A
INC DPTR
INC R1
DJNZ R2, G1
;接着送下16字节
MOV R2, #10H ;控制送数个数
MOV DPTR, #TAB2
PUSH 06H
I2: INC DPTR ;加上偏移量
DJNZ R6, I2
POP 06H
G2: CLR A
MOVC A, @A+DPTR
MOV @R1, A
INC DPTR
INC R1
DJNZ R2, G2
;完成出栈后返回
POP 03H
POP 02H
POP 01H
INC R6
RET
;=============================
;子程序名:延时约0.5ms
;51单片机频率为12MHz
;时钟周期为1/12M s
;一个机器周期等于12个时钟周期
;所以一个机器周期为1us
;想要延时5ms就需要执行500条指令
;需要修改延时时间只要修改R7和R6即可
;=============================
DELAY5: PUSH 07H
PUSH 06H
MOV R6, #2
LAB1: MOV R7, #250
LAB2: DJNZ R7, LAB2
DJNZ R6, LAB1
POP 06H
POP 07H
RET ;子程序返回
;================================
;字位码表,用于控制列选通
;只有选通的那一列是低电平
;其他列都是高电平
;这些数值都要送入P2口进行字位选通信号
;=================================
;字位码一直是从0-16,所以不用存了
;=============
;这里储存字形码
;=============
ORG 1000H
;这是上半部分
TAB1: db 08h, 28h, 48h, 88h, 68h, 18h, 00h,0FCh ;"鸡"
db 06h, 15h, 44h, 84h, 7Eh, 04h, 00h, 00h
db 40h, 20h,0F8h, 07h, 40h, 20h, 18h, 0Fh ;"你"
db 08h,0C8h, 08h, 08h, 28h, 18h, 00h, 00h
db 20h, 20h, 20h, 20h, 20h, 20h, 20h,0FFh ;"太"
db 20h, 20h, 20h, 20h, 20h, 30h, 20h, 00h
db 80h, 88h,0A8h,0A8h,0A9h,0AAh,0AEh,0F8h ;"美"
db 0ACh,0AAh,0ABh,0A8h,0ACh, 88h, 80h, 00h
;这是中间需要预留的一部分
DB 00H, 00H, 00H, 00H, 00H, 00H, 00H, 00H
DB 00H, 00H, 00H, 00H, 00H, 00H, 00H, 00H
;这是下半部分
TAB2: db 10h, 08h, 06h, 01h, 02h, 14h, 10h, 13h ;"鸡"
db 12h, 12h, 1Ah, 52h, 82h, 7Fh, 02h, 00h
db 00h, 00h,0FFh, 00h, 00h, 08h, 04h, 43h ;"你"
db 80h, 7Fh, 00h, 01h, 06h, 0Ch, 00h, 00h
db 40h, 40h, 20h, 20h, 10h, 0Ch, 0Bh, 30h ;"太"
db 03h, 0Ch, 10h, 10h, 20h, 60h, 20h, 00h
db 80h, 84h, 84h, 44h, 44h, 24h, 14h, 0Fh ;"美"
db 14h, 24h, 24h, 44h, 46h,0C4h, 40h, 00h
END