本文主要是介绍十二、分段程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
1. 使用db、dw、dd以及dup伪指令定义一段连续的数据空间:
1) 对于前三个伪指令,前缀d表示define,而后面的三个字母分别表示byte、word、double word的意思,分别用于定义字节空间、字空间、双字空间;
2) 示例:
assume cs:codesgcodesg segmentdb 0, 1, 2, 3dw 0, 1, 2, 3dd 0, 1, 2, 3mov ax, 4C00Hint 21H
codesg endsend
!注意:该系列伪指令都是在编译阶段就已经写入,因此运行程序之前(就是在程序加载进内存的时候)这些定义好的数据就已经存在于程序的内存空间中了,因此这里不需要单步调试来观察;
3) dup是duplicate(复制、拷贝的意思)的缩写,即如果想要使用上面的伪指令定义多个重复的值时可以使用dup伪指令,使用方法是:db/dw/dd 重复次数 dup(重复值列表)
请看示例:
assume cs:codesgcodesg segmentdb 3 dup(2) ; == 2 2 2 dw 2 dup(7, 9) ; == 7 9 7 9dd 3 dup(3, 1) ; == 3 1 3 1 3 1mov ax, 4C00Hint 21H
codesg endsend
4) 也可以定义字符(一个字符就是一个字节的ASCII码):
assume cs:codesgcodesg segmentdb '1234', '7890'; dw '7777' ; Bad! Character can only defined by db!db 3 dup('123', 'Abc') ; == '123Abc123Abc123Abc'mov ax, 4C00Hint 21H
codesg endsend
可以看到:
*1. 字符以及字符串必须都用单引号' '括起来;
*2. 字符型数据只能用db定义,用其它定义会直接编译报错!
2. 分段程序示例:
定义一个数据段和一个栈,然后利用栈将数据段中的内容的顺序反转(以字为单位):
assume cs:codesg, ds:datasg, ss:stacksgdatasg segmentdw 1234H, 5678H, 9ABCH, 0DEF0H
datasg endsstacksg segmentdw 10 dup(0)
stacksg endscodesg segmentdd 20 dup(0)start:mov ax, datasgmov ds, axmov ax, stacksgmov ss, axmov sp, 20mov bx, 0mov cx, 4
lp1:push [bx]add bx, 2loop lp1mov bx, 0mov cx, 4
lp2:pop [bx]add bx, 2loop lp2mov ax, 4C00Hint 21H
codesg endsend start
!注意:
i. 首先是标号start的作用,由于使用assume cs:codesg可以使程序在装载后cs指向codesg,但是这里也看到了,可以在代码段的开始处定义一段数据区域,因此在这种情况下cs:ip会指向这段数据区的开始处,如果直接从这里执行的话必然会造成程序出错,但是只要在最后end伪指令后指定一个标号并且将该标号作为程序真正的入口处地址就可以使程序在装载之后cs:ip指向该标号地址了,通常我们将该表好取名为start;
小结:我们可以利用end伪指令指定程序的入口处地址!在以上这段程序中codesg开始处的数据定义只是为了演示这个道理而已并没有实际意义;
ii. 我们可以利用类似定义codesg的方式定义其它多个段,但不过能用的也就最都只有4个段了,即cs、ds、ss、es;
iii. 但是assume只能真正改变cs:ip的指向,但是不能改变其它段寄存器的指向,因此其它段寄存器的指向需要在程序中手工指定!(这在前面讲到过)
*此程序的运行结果:
可以看到内容被反转了!
3. 段的物理大小:
1) 程序加载后是以16个字节为单位分配空间的,因此如果程序的实际内容刚好是16字节的整数倍则刚好,如果不是16的整数倍,则余数部分必须补全16的字节;
2) 具体说就是,如果段中的实际内容为N个字节,则需分两种情况讨论:
i. N为16的倍数,则物理大小就是N字节;
ii. 如果N不为16的倍数,则物理大小就是 ( N / 16 + 1 ) * 16;
iii. 因此将两种情况合并起来得到一个归一化的公式就是 ( ( N + 15 ) / 16 ) * 16;
4. 示例:定义三个段,将前两个段中的单元依次对应相加存入第三个段中对应的单元(字节单元)
assume cs:csegaseg segmentdb 1, 2, 3, 4, 5, 6, 7
aseg endsbseg segmentdb 7, 6, 5, 4, 3, 2, 1
bseg endscseg segmentdb 7 dup(9)start:mov ax, asegmov ss, axmov ax, bsegmov es, axmov ax, csegmov ds, axmov bx, 0mov cx, 7
lp:mov al, ss:[bx]mov [bx], almov al, es:[bx]add [bx], alinc bxloop lpmov ax, 4C00Hint 21H
cseg endsend start
!注意:可以看到除了代码段其它的段可以不用assume关联也可以随意定义;
运行结果:
5. 示例:将数据段中前8个字逆序存放在栈段的前8个字中
很简单,只需要push就行了,因为栈本来就是逆序扩张的
assume cs:codesg, ds:datasg, ss:stacksgdatasg segmentdw 1, 2, 3, 4, 5, 6, 7, 8
datasg endsstacksg segmentdw 9 dup(0)
stacksg endscodesg segmentstart:mov ax, datasgmov ds, axmov ax, stacksgmov ss, axmov bx, 0mov sp, 16mov cx, 8
lp:push [bx]add bx, 2loop lpmov ax, 4C00Hint 21H
codesg endsend start
运行结果:
这篇关于十二、分段程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!