单片机的查表程序
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
单片机的查表程序(共5页)
--本页仅作为文档封面,使用时请直接删除即可--
--内页可以根据需求调整合适字体及大小--
在单片机开发过程中.一些非线性的控制过程.最适合做一个表格来.时时改变系统的参数.达到控制的目的.最常的如产生正弦的的程.就是建一个大的数组时时改变输出的8位字节送给外部DA.由DA生成一个完整的正弦波.当然了.LED显示也是一个例子.通过建一个数组来实现段码的点亮点灭.下面就是一个LED表---digits[0]
#define SEG_a 0x01
#define SEG_b 0x02
#define SEG_c 0x04
#define SEG_d 0x08
#define SEG_e 0x10
#define SEG_f 0x20
#define SEG_g 0x40
#define SEG_dot 0x80
unsigned char digits[10] = {
(SEG_a|SEG_b|SEG_c|SEG_d|SEG_e|SEG_f), 过MCS-51指令系统中有专用的查表指令:MOVC A,@A+DP TR和MOV A,@A+PC.
MOVC A,@A+DPTR指令,DPTR作为基址寄存器时,其值为16位而且可根据需要设计,故可用于在64K ROM范围内查表。
编写查表程序时,首先把表的首址送入DPTR中,再将要查找的数据序号(或下表值)送入A中,然后就可以使用该指令进行查表操作,并将结果送累加器A中。
MOVC A,@A+PC指令,PC作为基址寄存器时,其值由指令的位置确定,它只能设在查表指令操作码下的256个字节范围内。
编写查表程序时,首先把查表数据的序号送入A中,再把从查表指令到表的首地址间的偏移量与A值相加,然后使用该指令进行查表操作,并把结果送累加器A中。
下面是把内部RAM 30H-37H单元中的十六进制数依次转换为ASCII码,并存入内部RAM 60H-6FH单元之中。
用查表法编写程序。
分析:由于十六进制数是从0~F,对应的ASCII码为30H~46H,每一个单元存放的16进制数,转换为ASCII码后分别存入2个单元,低位存低地址,高位存高地址。
用ANL A ,#0FH 分别取高4位和低4位。
进行查表转换成相应的ASCII码。
MOVC A , @A+DPTR 程序如下:
ORG 0000H
AJMP MAININITIAL
ORG 0050H
MAININITIAL: ;给30H-37H赋初值
MOV R0,#30H ;设置存数指针R0初值
MOV R1,#00H ;设置赋值变量个数计数寄存器R1(循环计数器)初值MOV DPTR,#initialtab
NEXTINITIAL:
MOV A,R1
MOVC A,@A+DPTR ;查表数据送累加器A
MOV @R0,A
INC R0 ;指针增一
INC R1 ;循环计数器增一
CJNE R1,#8,NEXTINITIAL ;判给30H-3FH赋初值完否
MAIN:
MOV R0,#30H ;设置十六进制数地址指针
MOV R1,#60H ;设置ASCII码地址指针
MOV R7,#08H ;需拼装的十六进制数字节个数
MOV DPTR,#ACSIITAB
ABC:
MOV A,@R0 ;取十六进制数
MOV B,A ;暂存
ANL A,#0F0H ;取十六进制数的字节高4位
RR A ;取十六进制数的字节高4位移到字节的低4位
RR A
RR A
RR A
MOVC A,@A+DPTR ;查表数据送累加器A
MOV @R1,A ;保存转换结果
INC R1 ;转换结果指针增一
MOV A,B
ANL A,#0FH ;取十六进制数的字节低4位
MOVC A,@A+DPTR ;查表数据送累加器A
MOV @R1,A ;保存转换结果
INC R1 ;转换结果指针增一
INC R0 ;转换数据指针增一
DJNZ R7,ABC ;继续 SJMP $
initialtab: ;给30H-37H赋值用初值表 DB 23H,6DH,09H,7FH,8CH,1EH,4BH,5AH ACSIITAB: ;0~F,对应的ASCII码表 DB "0" DB "9ABCDEF"
END
MOVC A,@A+PC程序如下:
ORG 0000H
AJMP MAININITIAL
ORG 0050H
MAININITIAL: ;给30H-37H赋初值
MOV R0,#30H ;设置存数指针R0初值
MOV R1,#00H ;设置赋值变量个数计数寄存器R1(循环计数器)初值MOV DPTR,#initialtab
NEXTINITIAL:
MOV A,R1
MOVC A,@A+DPTR ;查表数据送累加器A
MOV @R0,A
INC R0 ;指针增一
INC R1 ;循环计数器增一
CJNE R1,#8,NEXTINITIAL ;判给30H-3FH赋初值完否
MAIN:
MOV R0,#30H ;设置十六进制数地址指针
MOV R1,#60H ;设置ASCII码地址指针
MOV R7,#08H ;需拼装的十六进制数字节个数
MOV DPTR,#ACSIITAB
ABC:
MOV A,@R0 ;取十六进制数
MOV B,A ;暂存
ANL A,#0F0H ;取十六进制数的字节高4位
RR A ;取十六进制数的字节高4位移到字节的低4位
RR A
RR A
RR A
ACALL TRANACSAII ;查表数据送累加器A
MOV @R1,A ;保存转换结果
INC R1 ;转换结果指针增一
MOV A,B
ANL A,#0FH ;取十六进制数的字节低4位
ACALL TRANACSAII ;查表数据送累加器A
MOV @R1,A ;保存转换结果
INC R1 ;转换结果指针增一
INC R0 ;转换数据指针增一
DJNZ R7,ABC ;继续 SJMP $
initialtab: ;给30H-37H赋值用初值表 DB 23H,6DH,09H,7FH,8CH,1EH,4BH,5AH TRANACSAII:
INC A ;查表之前A加1是因为MOVC指令与数据表之间有一个地址单元的间隔(RET指令) MOVC A,@A+PC ;由于数据表紧跟MOVC指令之后,因此以PC作为基址寄存器比较方便. RET
ACSIITAB: ;0~F,对应的ASCII码表 DB "0" DB "9ABCDEF"
end
其它单片机跟C51查表类似.下面介绍松翰的单片机的查表程序.松翰单片机查表是通过寄存器来达到指向的表的目的如下
b0mov z,#segtab$L b0mov y,#segtab$M b0mov a,freqh ;led 3
add z,a
b0bts0 fc
incms y;表加1 nop movc
.................
segtab:
dw 000ch dw 000ah dw 000ch ;0 dw 000ch dw 0000h dw 0000h ;1 dw 0008h dw 000eh dw 0004h ;
2 ......................
希望看了此文之后.大家对查表的一个清楚认识。