在单片机开发过程中.一些非线性的控制过程.最适合做一个表格来.时时改变系统的参数.达到控制的目的.最常的如产生正弦的的程.就是建一个大的数组时时改变输出的8位字节送给外部DA.由DA生成一个完整的正弦波.当然了.LED显示也是一个例子.通过建一个数组来实现段码的点亮点灭.下面就是一个LED表---digits[0]#defineSEG_a0x01#defineSEG_b0x02#defineSEG_c0x04#defineSEG_d0x08#defineSEG_e0x10#defineSEG_f0x20#defineSEG_g0x40#defineSEG_dot0x80unsignedchardigits[10]={(SEG_a|SEG_b|SEG_c|SEG_d|SEG_e|SEG_f),//0(SEG_b|SEG_c),//1(SEG_a|SEG_b|SEG_d|SEG_e|SEG_g),//2(SEG_a|SEG_b|SEG_c|SEG_d|SEG_g),//3(SEG_b|SEG_c|SEG_c|SEG_f|SEG_g),//4(SEG_a|SEG_c|SEG_d|SEG_f|SEG_g),//5(SEG_a|SEG_c|SEG_d|SEG_e|SEG_f|SEG_g),//6(SEG_a|SEG_b|SEG_c),//7(SEG_a|SEG_b|SEG_c|SEG_d|SEG_e|SEG_f|SEG_g),//8(SEG_a|SEG_b|SEG_c|SEG_d|SEG_f|SEG_g)//9};C查表就太简单了temp2=digits[show_data[i]];一句搞定,C中还有一个switch语句也是一个很好的用查表语句C51汇编就相对麻烦一点.不过MCS-51指令系统中有专用的查表指令:MOVCA,@A+DPTR和MOVA,@A+PC.MOVCA,@A+DPTR指令,DPTR作为基址寄存器时,其值为16位而且可根据需要设计,故可用于在64KROM范围内查表。编写查表程序时,首先把表的首址送入DPTR中,再将要查找的数据序号(或下表值)送入A中,然后就可以使用该指令进行查表操作,并将结果送累加器A中。MOVCA,@A+PC指令,PC作为基址寄存器时,其值由指令的位置确定,它只能设在查表指令操作码下的256个字节范围内。编写查表程序时,首先把查表数据的序号送入A中,再把从查表指令到表的首地址间的偏移量与A值相加,然后使用该指令进行查表操作,并把结果送累加器A中。下面是把内部RAM30H-37H单元中的十六进制数依次转换为ASCII码,并存入内部RAM60H-6FH单元之中。用查表法编写程序。分析:由于十六进制数是从0~F,对应的ASCII码为30H~46H,每一个单元存放的16进制数,转换为ASCII码后分别存入2个单元,低位存低地址,高位存高地址。用ANLA,#0FH分别取高4位和低4位。进行查表转换成相应的ASCII码。MOVCA,@A+DPTR程序如下:ORG0000HAJMPMAININITIALORG0050HMAININITIAL:;给30H-37H赋初值MOVR0,#30H;设置存数指针R0初值MOVR1,#00H;设置赋值变量个数计数寄存器R1(循环计数器)初值MOVDPTR,#initialtabNEXTINITIAL:MOVA,R1MOVCA,@A+DPTR;查表数据送累加器AMOV@R0,AINCR0;指针增一INCR1;循环计数器增一CJNER1,#8,NEXTINITIAL;判给30H-3FH赋初值完否MAIN:MOVR0,#30H;设置十六进制数地址指针MOVR1,#60H;设置ASCII码地址指针MOVR7,#08H;需拼装的十六进制数字节个数MOVDPTR,#ACSIITABABC:MOVA,@R0;取十六进制数MOVB,A;暂存ANLA,#0F0H;取十六进制数的字节高4位RRA;取十六进制数的字节高4位移到字节的低4位RRARRARRAMOVCA,@A+DPTR;查表数据送累加器AMOV@R1,A;保存转换结果INCR1;转换结果指针增一MOVA,BANLA,#0FH;取十六进制数的字节低4位MOVCA,@A+DPTR;查表数据送累加器AMOV@R1,A;保存转换结果INCR1;转换结果指针增一INCR0;转换数据指针增一DJNZR7,ABC;继续SJMP$initialtab:;给30H-37H赋值用初值表DB23H,6DH,09H,7FH,8CH,1EH,4BH,5AHACSIITAB:;0~F,对应的ASCII码表DB012345678DB9ABCDEFENDMOVCA,@A+PC程序如下:ORG0000HAJMPMAININITIALORG0050HMAININITIAL:;给30H-37H赋初值MOVR0,#30H;设置存数指针R0初值MOVR1,#00H;设置赋值变量个数计数寄存器R1(循环计数器)初值MOVDPTR,#initialtabNEXTINITIAL:MOVA,R1MOVCA,@A+DPTR;查表数据送累加器AMOV@R0,AINCR0;指针增一INCR1;循环计数器增一CJNER1,#8,NEXTINITIAL;判给30H-3FH赋初值完否MAIN:MOVR0,#30H;设置十六进制数地址指针MOVR1,#60H;设置ASCII码地址指针MOVR7,#08H;需拼装的十六进制数字节个数MOVDPTR,#ACSIITABABC:MOVA,@R0;取十六进制数MOVB,A;暂存ANLA,#0F0H;取十六进制数的字节高4位RRA;取十六进制数的字节高4位移到字节的低4位RRARRARRAACALLTRANACSAII;查表数据送累加器AMOV@R1,A;保存转换结果INCR1;转换结果指针增一MOVA,BANLA,#0FH;取十六进制数的字节低4位ACALLTRANACSAII;查表数据送累加器AMOV@R1,A;保存转换结果INCR1;转换结果指针增一INCR0;转换数据指针增一DJNZR7,ABC;继续SJMP$initialtab:;给30H-37H赋值用初值表DB23H,6DH,09H,7FH,8CH,1EH,4BH,5AHTRANACSAII:INCA;查表之前A加1是因为MOVC指令与数据表之间有一个地址单元的间隔(RET指令)MOVCA,@A+PC;由于数据表紧跟MOVC指令之后,因此以PC作为基址寄存器比较方便.RETACSIITAB:;0~F,对应的ASCII码表DB012345678DB9ABCDEFend其它单片机跟C51查表类似.下面介绍松翰的单片机的查表程序.松翰单片机查表是通过X.Y寄存器来达到指向的表的目的如下b0movz,#segtab$Lb0movy,#segtab$Mb0mova,freqh;led3addz,ab0bts0fcincmsy;表加1nopmovc.................segtab:dw000chdw000ahdw000ch;0dw000chdw0000hdw0000h;1dw0008hdw000ehdw0004h;2......................希望看了此文之后.大家对查表的一个清楚认识