简介 :
子程序描述 :
名称 : show_str
功能 : 在指定的位置显示一个用 0 结束的字符串
参数 : (dh) = 行号 (0-24) , (dl) = 列号 (0-79)
返回 : None
分析 :
根据之前的分析 , 我们已经了解了显存的范围等等信息
也知道了只要往显存中写入数据 , 就可以显示到对应的位置
我们这里需要做的一件事就是搞清楚显存地址和行号列号的关系即可
1. 显存地址 : B8000H - BFFFFH
2. 每一个字符占用两个字节 , 低地址为 ASCII 码 , 高地址为属性
可以计算出 , 假设我们要往显存第 n 页的第 x 行 第 y 列写入 ascii 码为 0x20 的字符 , 属性为 : 11001000B (高亮闪烁黑底红字)
3. 为了能对齐 , 我们将一页的数据记为 80 * 25 * 2 = 4000 字节 , 约为 4k
那么就应该将 0x20 写到 : B8000H + (n - 1) * (4 * 1024) + (x - 1) * (80) + (y - 1) 处
将属性 11001000B 写到 : B8000H + (n - 1) * (4 * 1024) + (x - 1) * (80) + (y - 1) + 1 处
我们这里需要将数据写入第一页的 dh 行 dl 列 , 假设调用的时候第一行即用 (dh) = 0 来表示
也就是说 :
(B800H:0000H + dh * 80 + dl) = ASCII
(B800H:0000H + dh * 80 + dl + 1) = ATTR
代码 :
assume cs:code,ds:data,ss:stack
data segment
db "Freedom", 0
data ends
stack segment
db 16 dup(0)
stack ends
code segment
start:
; 设置数据段
mov ax, data
mov ds, ax
; 设置栈段
mov ax, stack
mov ss, ax
mov sp, 16
; 设置显存段地址
mov ax, 0B800H
mov es, ax
mov si, 0 ; 数据段偏移地址 , 指向第一个字符
; 设置显示的位置
mov dh, 8
mov dl, 0
call show_str
jmp finish
; 功能 : 在屏幕指定位置显示一个指定字符串
; 参数 : dh 表示行 , dl 表示列 , ds 和 si 表示内存中字符串的首地址
; 返回 : None
show_str:
each_chr:
mov ch, 00H
mov cl, ds:[si]
jcxz over
mov ch, 11001000B ; 设置字符属性
call show_chr
inc dl ; 列数自增
inc si ; 字符指针自增
jmp each_chr
over:
ret
; 功能 : 在屏幕上的指定位置显示一个指定字符
; 参数 : dh 表示行 , dl 表示列 , cl 表示字符 , ch 表示字符的属性
; 返回 : None
show_chr:
; 计算相对显存地址的偏移
; (行数 * 80 + 列数) * 2 = 字节数
mov al, 80
mul dh
push dx
mov dh, 00H
add ax, dx
pop dx
add ax, ax
mov di, ax
mov es:[di], cx
ret
finish:
; 为了能看到效果 , 这里循环执行 nop
jmp stop
mov ax,4cH
int 21H
stop:
nop
jmp stop
code ends
end start
备注 :