zf标志:零标志位,zf=1代表结果为0,否则为1
mov ax,1
sub ax,1
sub结果为0,所以zf=1
注意:
在8086CPU的指令集中,有的指令的执行是影响标志寄存器的,比如:add、sub、mul、div、inc、or、and等,它们大都是运算指令(进行逻辑或算术运算)。
有的指令的执行对标志寄存器没有影响,比如:mov、push、pop等,它们大都是传送指令。
我们在使用一条指令的时候,要注意这条指令的全部功能,其中包括,执行结果对标记寄存器的哪些标志位造成影响。
pf标志:奇偶标志位
记录指令执行后,结果的所有二进制位中1的个数:为偶数,PF = 1;为奇数,PF = 0
例
mov al,1
add al,10
执行后,结果为00001011B,其中有3(奇数)个1,则PF=0;
sf标志位:符号标志位。SF = 1 结果为负;SF = 0 结果为正
(计算机中的一个数据可以看作是有符号数,也可以看成是无符号数)
如:00000001B ,可以看作为无符号数 1 ,或有符号数+1
10000001B ,可以看作为无符号数129,也可以看作有符号数-127
这也就是说,对于同一个二进制数据,计算机可以将它当作无符号数据来运算,也可以当作有符号数据来运算。
示例
mov al,10000001B
add al,1
结果:(al) = 10000010B
我们可以将add指令进行的运算当作无符号数的运算,那么add指令相当于计算129+1,结果为130(10000010B)
也可以将add指令进行的运算当作有符号数的运算,那么add指令相当于计算-127+1,结果为-126(10000010B)
不管我们如何看待,CPU 在执行add等指令的时候,就已经包含了两种含义,也将得到用同一种信息来记录的两种结果。
关键在于我们的程序需要哪一种结果。
终结:计算机的计算过程是固定的,该怎样就怎样,是否把数据看成补码是程序员的事情
cf标志:进位标志位。cf=1代表产生了进位或借位
of标志:溢出标志,OF=1代表溢出
adc指令:adc是带进位加法指令 ,它利用了CF位上记录的进位值。
adc 操作对象1=操作对象1+操作对象2+CF
示例:mov ax,2
mov bx,1
sub bx,ax
adc ax,l
结果为:ax = ax+1+cf = 4
示例二:mov ax,1
add ax,ax
adc ax,3
结果为:ax = ax+3+cf = 2+3+0 = 5
示例三:
mov al,98H
add al,al
adc al,3
分析:98H+98H = 130H = 100110000B 由于al寄存器只有8位所以:al = 00110000B = 30H
结果为:al = al+3+cf = 30+3+1 = 34H
编程:计算1EF000H+201000H 结果放在ax(高16位)和 bx(低16位)中。
上机计算(已知1EF000+201000=3F0000)
分析:把1EF000 分为两部分001E和F000分别存入ax,bx中;也把201000分为两部分:0010和1000 分别和ax,bx相加;由于bx中的计算可能会产生进位,所以ax需要加上进位信息cf
代码为:mov ax,001EH
mov bx,0F000H
add bx,1000H
adc ax,0020H
结果为:ax 003F;bx 0000;
sbb指令:
sbb ax,bx 执行为:ax = ax-bx-CF
例子:
计算算003E100OH–00202000H,结果放在ax,bx中,程序如下:
mov bx,1000H
mov ax,003EH
sub bx,2000H
sbb ax,0020H
cmp指令:
cmp 是比较指令,功能相当于减法指令,只是不保存结果。
cmp 指令执行后,将对标志寄存器产生影响。
例:mov ax,8
mov bx,3
cmp ax,bx
执行后:
(ax) = 8
ZF = 0,
PF = 1,
SF = 0,
CF = 0,
OF = 0。
cmp可以配合转移指令来使用,cmp使标志寄存器发生改变,转移指令根据标准寄存器来判断大小
如:cmp ah,bh
je s
add ah,bh
jmp short ok
s:add ah,ah
ok:mov ax,4c00h
int 21h //实现功能如果ah==bn则ah=ah+ah,否则ah=ah+bh
DF标志位:方向标志位,在串处理指令中,控制每次操作后si,di的增减。
DF = 0:每次操作后si,di递增
DF = 1:每次操作后si,di递减
DF、movsb、movsw
movsb:以字节为单位将 ds:si 指向的内存单元中的字节送入 es:di中,然后根据标志寄存器DF位的值,将 si和di递增或递减。
movsb == ((es)*16+(di))=((ds)*16+(si)),如果df=0则:si++,di++反之亦然
movsw:以字为单位将 ds:si 指向的内存单元中的字送入 es:di中,然后根据标志寄存器DF位的值,将 si和di递增2或递减2。
rep使用格式如下:rep movsb == s:movsb,loop s
rep的作用是根据cx的值,重复执行后面的串传送指令。
由于每执行一次movsb指令si和di都会递增或递减指向后一个单元或前个单元,则rep movsb就可以循环实现(cx)个字符的传送。
由于flag的DF位决定着串传送指令执行后,si和di改变的方向,所以CPU应该提供相应的指令来对DF位进行设置,从而使程序员能够决定传送的方向。
8086CPU提供下而两条指令对DF位进行设置:
cld指令:将标志寄存器的DF位置0
std指令:将标志寄存器的DF位置1
例:将下列第一个数据段复制到第二个数据段
data segment
db 'Welcome to mass!'
db 16 dup(0)
data ends
解:start: mov ax,data
mov ds,ax
mov si,0
mov es,ax
mov di,10H
mov cx,16
cld
rep movsb