引言
这一章,我们讨论如何有效合理地组织数据,以及相关地编程技术。
本章中,我们要用到这种标号,先进行如下介绍
前面地课程中,我们一直在代码段中使用标号来标记指令、数据、段的起始地址。
比如:下面的程序将code段中的a标号处的8个数据累加,结果存储到b标号处的字中。
程序中,code、a、b、start、s都是标号。这些标号仅仅表示了内存单元的地址。
但是,我们还可以使用一种标号,这种标号不但表示内存单元的地址,还表示了内存单元的长度,即表示在此标号处的单元,是一个字节单元,还是字单元,还是双字单元。
上面的程序我们还可以写成这样:
我们在code段中使用的标号a、b后面没有:,他们是同时描述内存地址和单元长度的标号。
标号a,描述了地址code:0,和从这个地址开始,以后的内存单元都是字节单元;
而标号b描述了地址code:8,和从这个地址开始,以后的内存单元都是字单元。
因为这种标号包含了对单元长度的描述,所以,在指令中,它可以代表一个段中的内存单元。
比如,对于程序中的b dw 0
指令 :mov ax,b
相当于:mov ax,cs:[8]
指令:mov b,2
相当于:mov word ptr cs:[8],2
指令:inc b
相当于: inc word ptr cs:[8]
在这些指令中,标号b代表了一个内存单元,地址为code:8,长度为2字节。
下面的指令会引起编译错误:
mov al,b
因为b代表的内存单元是字单元,而al是8位寄存器。
如果我们将程序中的指令:add b,ax,写为add b,al
将出现同样的编译错误。
可见,使用这种包含单元长度的标号,可以使我们以简洁的形式访问内存中的数据。
以后,我们将这种标号称为数据标号。
它标记了存储数据的单元的地址和长度。
它不同于仅仅表示地址的地址标号。
16.2 在其他段中使用数据标号
一般来说,我们不再代码段中定义数据,而是将数据定义到其他段中。
在其他段中,我们也可以使用数据标号来描述存储数据的单元的地址和长度。
注意:在后面加有:的地址标号,只能在代码段中使用,不能在其他段中使用。
注意,如果想在代码段中,直接用数据标号访问数据,则需要用伪指令assume将标号所在的段和一个段寄存器联系起来。
否则编译器在编译的时候,无法确定标号的段地址在哪一个寄存器中。
当然,这种联系是编译器需要的,但绝对不是说,我们因为编译器的工作需要,用assume指令将段寄存器和某个段相联系,段寄存器中就会真的存放该段的地址。
我们在程序中还要使用指令对段寄存器进行设置。
比如:在上面的程序中,我们要在断码段code中用data段中的数据标号a、b访问数据,则必须用assume将一个寄存器和data段相联。
在程序中,我们用ds寄存器和data段相联,则编译器对相关指令的编译如下:
指令:mov al,a[si]
编译为:mov al,[si+0]
指令:add b,ax
编译为: add [8],ax
因为这些实际编译出的指令,都默认所访问单元的段地址在ds中,而实际要访问的段为data,所以,若要访问正确,在这些指令执行前,ds中必须为data段的段地址。
我们可以将标号当作数据来定义,此时,编译器将标号所表示的地址当作数据的值。
比如:
data segment
a db 1,2,3,4,5,6,7,8
b dw 0
c dw a,b
data ends
数据标号c处存储的两个字型数据为标号a、b的偏移地址。
数据标号c处存储的两个双字型数据为标号a的偏移地址和段地址、标号b的偏移地址和段地址。
16.3 直接定址表
现在,我们讨论用查表得方法编写相关程序得技巧。
编写子程序 ,以十六进制得形式在屏幕中间显示给定得byte型数据。
分析:
一个字节需要两个十六进制数码来表示,所以,子程序需要在屏幕上显示两个ASCII字符。
我们可以将一个byte得高4位和低4位分开,分别用他们得值得到对应的数码字符。
比如2bh,我们可以得到高4位的值为2,低4位的值为11.
那么我们如何用这两个数值得到对应的数码字符2和b呢
更简洁的做法是:
我们建立一张表,表中依次存储字符0~f,我们可以通过数值0~15直接查找到对应的字符。
assume cs:code code segment start: mov al,0eh call showbyte mov ax,4c00h int 21h ;子程序: ;用al传送要显示的数据 showbyte: jmp short show table db '0123456789ABCDEF' ;字符表 show: push bx push es mov ah,al shr ah,1 shr ah,1 shr ah,1 shr ah,1 ;右移4位,ah中得到高4位的值 and al,00001111b ;al中为低4位的值 mov bl,ah mov bh,0 mov ah,table[bx] ;用高4位的值作为相对于table的偏移,取得对应的字符 mov bx,0b800h mov es,bx mov es:[160*12+40*2],ah mov bl,al mov bh,0 mov al,table[bx] ;用低4位的值作为相对于table的偏移,取得对应的字符 mov es:[160*12+40*2+2],al pop es pop bx ret code ends end start
利用表,在两个数据集合之间建立一种映射关系,使我们可以用查表得方法根据给出得数据得到其在另一集合中得对应数据。
这样做得目的一般来说有三个:
1)为了算法得清晰和简洁
2)为了加快运算速度
3)为了使程序易于扩充
上面得子程序中,体现得更多的是算法得清晰和简洁,下面这个例子是为了加快运算速度而采用的查表得方法。
assume cs:code code segment start: mov al,60 call showsin mov ax,4c00h int 21h showsin: jmp short show table dw ag0,ag30,ag60,ag90,ag120,ag150,ag180 ;字符串偏移地址表 ag0 db '0',0 ;sin(0)对应的字符串“0” ag30 db '0.5',0 ;sin(0)对应的字符串“0.5” ag60 db '0.866',0 ;sin(0)对应的字符串“0.866” ag90 db '1',0 ;sin(0)对应的字符串“1” ag120 db '0.866',0 ;sin(0)对应的字符串“0.866” ag150 db '0.5',0 ;sin(0)对应的字符串“0.5” ag180 db '0',0 ;sin(0)对应的字符串“0” show: push bx push es push si mov bx,0b800h mov es,bx ;以下用角度值/30 作为相对于table的偏移量,取得对应的字符串的偏移地址,放在bx中 mov ah,0 mov bl,30 div bl mov bl,al mov bh,0 add bx,bx mov bx,table[bx] ;以下显示sin(x)对应的字符串 mov si,160*12+40*2 shows: mov ah,cs:[bx] cmp ah,0 je showret mov es:[si],ah inc bx add si,2 jmp shows showret: pop si pop es pop bx ret code ends end start
下面,我们讨论一下各种功能如何实现:
1)清屏:将显存中当前屏幕中的字符设为空格符。
2)设置前景色:设置显存中当前屏幕中处于奇地址的属性字节的第0、1、2位
3)设置背景色:设置显存中当前屏幕中处于奇地址的属性字节的第4、5、6位。
4)向上滚动一行:依次将第n+1行的内容复制到第n行处:最后一行为空。
;编程:实现一个子程序setscreen,为显示输出提供如下功能: ;(1) 清屏。 ;(2) 设置前景色。 ;(3) 设置背景色。 ;(4) 向上滚动一行。 ; ;入口参数说明: ;(1) 用 ah 寄存器传递功能号:0 表示清屏,1表示设置前景色,2 表示设置背景色,3 表示向上滚动一行; ;(2) 对于2、3号功能,用 al 传送颜色值,(al) ∈{0,1,2,3,4,5,6,7} setscreen: jmp short set table dw sub1,sub2,sub3,sub4 set: push bx cmp ah,3 ;判断传递的是否大于 3 ja sret mov bl,ah mov bh,0 add bx,bx ;根据ah中的功能号计算对应子程序的地址在table表中的偏移 call word ptr table[bx] ;调用对应的功能子程序 sret: pop bx iret ;功能子程序1:清屏 sub1: push bx push cx push es mov bx,0b800h mov es,bx mov bx,0 mov cx,2000 sub1s: mov byte ptr es:[bx],' ' add bx,2 loop sub1s pop es pop cx pop bx ret ;sub1 ends ;功能子程序2:设置前景色 sub2: push bx push cx push es mov bx,0b800h mov es,bx mov bx,1 mov cx,2000 sub2s: and byte ptr es:[bx],11111000b or es:[bx],al add bx,2 loop sub2s pop es pop cx pop bx ret ;sub2 ends ;功能子程序3:设置背景色 sub3: push bx push cx push es mov cl,4 shl al,cl mov bx,0b800h mov es,bx mov bx,1 mov cx,2000 sub3s: and byte ptr es:[bx],10001111b or es:[bx],al add bx,2 loop sub2s pop es pop cx pop bx ret ; sub3 ends ;功能子程序4:向上滚动一行 sub4: push cx push si push di push es push ds mov si,0b800h mov es,si mov ds,si mov si,160 ;ds:si指向第n+1行 mov di,0 ;es:di指向第n行 cld mov cx,24;共复制24行 sub4s: push cx mov cx,160 rep movsb ;复制 pop cx loop sub4s mov cx,80 mov si,0 sub4s1: mov byte ptr es:[160*24+si],' ' ;最后一行清空 add si,2 loop sub4s1 pop ds pop es pop di pop si pop cx ret ;sub4 ends