本系列将讲解《汇编语言》一书,本节讲解**【汇编语言】16-直接定址表**。
本期源码,欢迎star:
https://github.com/ZYKWLJ/assembly4-homework
| 本节速览 |
|---|
| 1.地址标签是?有何特点?怎么使用?给出例程? |
| 2.数据标签是?有何特点?怎么使用?给出例程? |
| 3.我们的cs、start、ds等标签是地址标签还是数据标签? |
| 4.地址标签和数据标签在形式上、具体含义上有何不同? |
| 5.我们使用数据标签汇中的数据,需要有哪些改动?assume+ds段赋值(如果是ds段)。 |
一、描述了单元长度的标号
1、两段程序的对比

1.普通程序-使用地址标签定义
; 体会`a:`和`a`的区别
; 本节讲述 `a:`,也就是段的标签,仅仅表示一个地址,对后面的数据长度不做确保
assume cs:code
code segment
str:
db 'a','b','c'
start:
mov ah, 9
mov si, offset str
mov al, cs:[si]
mov bl, 11001010b
mov bh, 0
mov cx, 1
int 10h
mov ax, 4c00h
int 21h
code ends
end start
这里使用str:标签(注意是带冒号的),来完成数据的定义,并且我们再后面引用其数据时,使用的是offset str,这得到了str地址标签的偏移量。
实际上,这里的start、cs都是和str一样的类型,因为他们后面可以跟各种指令、数,含义也仅表示了内存单元的地址。
2.数据标签-用来标签数据
; 体会`a:`和`a`的区别
; 本节讲述 `a`,也就是数据地址的标签,仅仅表示一个地址,对后面的数据长度做确保
assume cs:code
code segment
str db 'a','b','c';这后面的数据全是字节
start:
mov ah, 9
mov si, 0
mov al, str[si]
mov bl, 11001010b
mov bh, 0
mov cx, 1
int 10h
mov ax, 4c00h
int 21h
code ends
end start
注意,这里str后面没有了冒号!
在汇编中,这说明,str标签后面全是类型大小一致的数据,所以此时str也称为数据标签!。其实,这里的str表示的是code:0,并且从这个地址开始,后面的内存单元全是字节。
注意此时我们使用str数据的方式:str[si],没有冒号,直接是基地址+索引,很像C语言中的数组。
str[si]通过cs:0转化为cs:[si+0]也是一样的。
可见,使用这种包含单元长度的标号,可以使我们以简洁的形式访问内存中的数据。
以后尽量使用这种数据标签,!
2、将数据标签放在其他段中
我们在正式开发中,很少将数据混在代码段中,都是另开段专门存放。
源码如下:
assume cs:code ,ds:data
data segment
str db 'a','b','c'
data ends
code segment
start:
mov ax, data ;这两步是必要的,因为使得data地址标签和ds段寄存器产生关联
mov ds, ax
mov ah, 9
mov si, 0
mov al, str[si]
mov bl, 11001010b
mov bh, 0
mov cx, 1
int 10h
mov ax, 4c00h
int 21h
code ends
end start
值得注意的是,这里新增了如下两处变动:
assume ds:data
mov ax, data ;This is necessary because it establishes the association between the data address label and the ds segment register.
mov ds, ax
第一段变动是因为:
这是因为,如果想在代码段中直接用数据标号访问数据段汇中的数据,则需要用伪指令 assume 将数据标号所在的段和一个段寄存器联系起来。否则编译器在编译的时候,无法确定标号未来的段地址在哪一个寄存器中。
第二段变动是因为:
当然,前面知识告知编译器,下面的赋值才是真正地让数据标签的地址赋值给段寄存器
mov ax, data ;This is necessary because it establishes the association between the data address label and the ds segment register.
mov ds, ax
所以这两段缺一不可。
其次,在使用跨段数据时,必须使用数据标签,不能使用地址标签。 否则会报错,不信你试试。
二、直接定址表
本节,我们讨论用查表的方法编写相关程序的技巧。
0.定义
在已知产生的结果集(一般都是有限的)的情况下,直接根据输入数据,在表中查找对应结果。这样的表,称为直接定址表。
一般,我们使用直接定址表有两大极其明显的特点:
| 直接定址表的使用暗示与特点 |
|---|
| 数据有限 |
| 输入对应的输出关系已知且明确 |
下面,我们通过两道习题感受一下直接定址表的使用及威力。
1.便于程序清晰
1.题目
以十六进制的形式在屏幕中间显示给定的字节型数据。
即例如,11110010,显示为F2
2.分析
显然,题目的输入有限,并且输出也有限。并且输出输出对应关系明确且已知,那么这种显然使用直接定址表,大大简化程序复杂性。
题目等价于将字节数据的高低4位数据对应015`输出为`0F`。
3.程序
使用直接定址表,便捷地完成0-15的十六进制关系对应。
assume cs:code, ds:data
data segment
table db '0123456789ABCDEF' ; 字符表
data ends
code segment
start:
mov ax, data
mov ds, ax
; 测试显示不同字节
mov al, 1Bh ; 显示 "1B"
call showbyte
mov ax, 4c00h
int 21h
; 子程序:显示AL中的字节(十六进制)
showbyte:
push bx
push es
push ax
mov ah, al ; 保存原始值
mov cl, 4
shr ah, cl ; 右移4位获取高4位
and al, 00001111b ; 获取低4位
mov bx, 0b800h
mov es, bx
; 显示高4位(绿色)
mov bl, ah
mov bh, 0
mov dl, table[bx] ; 字符
mov dh, 02h ; 颜色属性(绿色)
mov es:[160*24], dx ;显示到下一行
; 显示低4位(红色)
mov bl, al
mov bh, 0
mov dl, table[bx] ; 字符
mov dh, 04h ; 颜色属性(红色)
mov es:[160*24+2], dx ;显示到下一行
pop ax
pop es
pop bx
ret
code ends
end start

可见,使用直接定址表,我们避免了大段大段的if-else、cmp、jne等指令,方便快捷优美。
2.加快运算速度
比如,我们要求sin10,sin39等等数据时,常常使用直接定址表,因为sin非特殊角正弦值运算较为复杂,运算耗时,而我们将结果直接保存,可大大减少运算时间。
一般将1~90的所有正弦值都保存下来,使用直接定址表!
核心思路已有,具体程序,不再展开!

| 本节速览 |
|---|
| 1.地址标签是?有何特点?怎么使用?给出例程? |
| 2.数据标签是?有何特点?怎么使用?给出例程? |
| 3.我们的cs、start、ds等标签是地址标签还是数据标签? |
| 4.地址标签和数据标签在形式上、具体含义上有何不同? |
| 5.我们使用数据标签汇中的数据,需要有哪些改动?assume+ds段赋值(如果是ds段)。 |
本系列将精讲Linux0.11内核中的每一个文件,共计会发布100+文章。
😉【Linux102】11-kernel/vsprintf.c
😉【Linux102】12-include/stdarg.h
😉【Linux102】14-kernel/system_call.s
😉【Linux102】15-include/linux/sched.h
😉【Linux102】18-include/signal.h
😉【Linux102】19-include/sys/types.h
😉【Linux102】20-include/linux/kernel.h
😉【Linux102】21-include/asm/segment.h
😉【Linux102】22-include/linux/head.h
😉【Linux102】23-include/linux/mm.h
😉【Linux102】24-include/linux/fs.h
😉【Linux102】26-include/sys/wait.h
😉【Linux102】27-include/inux/tty.h
😉【Linux102】28-include/termios.h
😉【Linux102】30-include/sys/times.h
😉【Linux102】31-include/sys/utsname.h
😉【Linux102】32-include/stddef.h
😉【Linux102】33-include/linux/sys.h
😉【Linux102】36-include/asm/system.h
😉【Linux102】38-include/linux/fdreg.h
😉【Linux102】39-include/asm/io.h
😉【Linux102】40-kernel/blk_drv/blk.h
😉【Linux102】41-kernel/blk_drv/hd.c
😉【Linux102】42-include/linux/config.h
😉【Linux102】43-include/linux/hdreg.h
😉【Linux102】45-kernel/blk_drv/ramdisk.c
😉【Linux102】46-include/asm/memory.h
😉【Linux102】47-include/string.h
😉【Linux102】48-kernel/blk_drv/floppy.c
😉【Linux102】49-kernel/chr_drv/keyboard.S
😉【Linux102】50-kernel/chr_drv/console.c
😉【Linux102】51-kernel/chr_drv/serial.c
😉【Linux102】52-kernel/chr_drv/rs_io.s
😉【Linux102】53-kernel/chr_drv/tty_io.c
😉【Linux102】56-kernel/chr_drv/tty_ioctl.c
和Linux内核102系列不同,本系列将会从全局描绘Linux内核的各个模块,而非逐行源码分析,适合想对Linux系统有宏观了解的家人阅读。
😉【Linux】Linux概述1-linux对物理内存的使用
本系列将带领大家从0开始循序渐进学习汇编语言,直至完全掌握这门底层语言。同时给出学习平台DOSBox的使用教程。
😉【汇编练习】12-验证ROM和其他内存区域的OS的接管情况
本系列将直击C语言的本质基础,流利处理出各个易错、实用的实战点,并非从零开始学习C。
关于小希
😉嘿嘿嘿,我是小希,专注Linux内核领域,同时讲解C语言、汇编等知识。
我的微信:C_Linux_Cloud,期待与您学习交流!

加微信请备注哦
小希的座右铭:
别看简单,简单也是难。别看难,难也是简单。我的文章都是讲述简单的知识,如果你喜欢这种风格:
下一期想看什么?在评论区留言吧!我们下期见!

1278

被折叠的 条评论
为什么被折叠?



