经过学习,对8086汇编常用指令进行了以下总结:
(1).mov : 移动数据
例如:
mov ax, 8h ;mov 寄存器,常量 mov ax, bx ;mov 寄存器,寄存器 mov ax, ds:[0] ;mov 寄存器,内存单元 mov ds:[0], ax ;mov 内存单元,寄存器 mov ds, ax ;mov 段寄存器,寄存器 mov word ptr ds:[0], 8h ;mov (内存单元大小声明) 内存单元, 常量 sk: mov ax, offset sk ;mov 寄存器,标号偏移地址 mov byte ptr ds:[0], offset sk ;mov 内存单元,标号偏移地址
易错点:
mov ds:[0], ds:[1] ;mov 内存单元,内存单元(非法指令) mov ds, 8h ;mov 段寄存器,常量(非法指令) mov ds, es ;mov 段寄存器,段寄存器(非法指令) mov ds, offset sk ;mov 段寄存器,标号偏移地址(非法指令) mov [0], ax ;常量偏移地址不可省略默认ds,必须作段地址声明(非法指令)
(2).add:增加数据值,用法同mov类似
(3).sub:减少数据值,用法同mov类似
(4).push:入栈,栈顶指针(sp)=(sp)+2,,并将一个字数据存入sp所指内存单元
例如:
push ax ;push 寄存器 push ds:[0] ;push 内存单元
易错点:
push al ;push 寄存器(高/低)字节(非法指令),必须是一个字,16位数据 push 8h ;push 常量(非法指令)
(5).pop:出栈,读取一个字sp所指的内存单元,并使栈顶指针(sp)=(sp)-2
例如:
pop ax ;pop 寄存器 pop ds:[0] ;pop 内存单元
易错点:
pop al ;pop 寄存器(高/低)字节(非法指令),必须是一个字,16位数据 pop 8h ;pop 常量(非法指令)
(6).inc:数据值+1, dec:数据值-1
例如:
inc ax ;inc 寄存器 inc byte ptr ds:[0] ;inc (大小声明)内存单元 dec ax ;dec 寄存器 dec byte ptr ds:[0] ;dec (大小声明)内存单元
易错点:
inc 8h ;inc 常量(非法指令) dec 8h ;dec 常量(非法指令)
(7).jmp:无条件转移指令
例如:
s: jmp short s ;段内短转移,jmp short 标号,IP修改范围-128~127 jmp near ptr s ;段内近转移,jmp near ptr 标号, IP修改范围-32768~32767 ;以上两条指令的实质为保存IP到标号的偏移距离,注意正确使用 jmp far ptr s ;段间转移(远转移),jmp far ptr s, 修改CS:IP为标号的段地址:偏移地址 jmp ax ;段内转移,jmp 16位reg, (IP)=(16位reg) jmp word ptr ds:[0] ;段内转移,jmp word ptr 内存单元地址, ;(IP)=(内存单元地址处双字节内存) jmp dword ptr ds:[0] ;段间转移,jmp dword ptr 内存单元地址, ;(IP)=(内存单元地址处低16位内存数据),(CS)=(内存单元地址处高16位内存数据)
易错点:
jmp 1000:0 ;想转移到(cs:ip)=(1000:0)处(非法指令) jmp offset s ;想转移到标号s处(非法指令)
(8).jcxz:有条件转移指令,等价于
if((cx)==0) jmp short 标号;
(9).loop:循环指令,等价于
(cx)--; if((cx)!=0)jmp short 标号;
(10).and:二进制与运算,同mov,add,sub用法相似
(11).or:二进制或运算, 同mov,add,sub用法相似
例如:
and ds:[0], 1111h ;and 内存单元 and ax, 1111h ;and 16位寄存器 常量 and al, 11111110b ;and 8位寄存器 常量(第0位设为0) and ax, bx ;and 16位寄存器,16位寄存器 ;等等... or ds:[0], 1111h ;or 内存单元 or ax, 1111h ;or 16位寄存器 常量 or al, 00000001b ;or 8位寄存器 常量(第0位设为0) or ax, bx ;or 16位寄存器,16位寄存器 ;等等...
(12).mul:乘法指令
;(1)两个相乘的数:两个相乘的数,要么都是8位,要么都是16位。如果是8位, ;一个默认放在AL中,另一个放在8位reg或内存字节单元中;如果是16位,一个默认在 ;AX中,另一个放在16位reg或内存单元中。 ;(2)结果:如果是8位乘法,结果默认放在AX中;如果是16位乘法,结果高位默认在 ;DX中存放,低位在AX中放。 ;计算100*10 mov al, 100 mov bl, 10 mul bl ;结果:(ax)=1000(03E8h) ;计算100*10000 mov ax, 100 mov bx, 10000 mul bx ;结果:(ax)=4240H, (dx)=000FH (F4240H=1000000)
(13).div:除法指令
;除数:有8位和16位两种,在一个reg或内存单元中 ;被除数:默认放在AX或DX和AX中,如果除数为8位,被除数则为16位, ;默认在AX中存放;如果除数为16位,被除数位32位,在DX和AX中存放,DX存 ;放高16位,AX存放低16位 ;结果:如果除数为8位,则AL存储除非操作的商,AH存储除非操作的余数; ;如果除数为16位,则AX存储除法操作的商,DX存储除法操作的余数 ;计算100001/100 mov dx, 1 mov ax, 86A1H ;(dx)*10000H+(ax)=100001 mov bx, 100 div bx ;结果:(ax)=03E8H(即1000),(dx)=1(余数为1) ;计算1001/100 mov ax, 1001 mov bl, 100 div bx ;结果:(al)=0AH(即10),(ah)=1(余数为1)
(14)call 标号:等价于
push IP jmp near ptr 标号
(15)ret : 等价于
pop IP
因此常如此配合使用:
call program1 program1: ;........ ret
(16)call far ptr 标号:等价于
push CS push IP jmp far ptr 标号
(17).retf:等价于
pop IP pop CS
因此常如此配合使用:
call far ptr program2 program2: ;........ retf
(18)call word ptr 内存地址:等价于
push IP jmp word ptr 内存地址 ;例如: mov sp, 10h mov ax, 0123h mov ds:[0], ax call word ptr ds:[0] ;结果:(IP)=0123H, (SP)=0EH
(19)call dword ptr 内存地址:等价于
push CS push IP jmp dword ptr 内存单元地址 ;例如: mov sp, 10h mov ax, 0123h mov ds:[0], ax mov word ptr ds:[2], 0 call dword ptr ds:[0] ;结果:(CS)=0,(IP)=0123H,(sp)=0CH
(20)shl:逻辑左移指令
;(1)将一个寄存器或内存单元中的数据向左移位; ;(2)将最后移出的一位写入CF中 ;(3)最低位用0补充 ;例如: mov al, 01001000b shl al, 1 ;结果:(al)=10010000b, CF=0 ;若移动位数大于1,则必须将移动位数放在cl中 mov al, 01010001b mov cl, 3 shl al, cl ;结果:(al)=10001000b, 因为最后移除的一位是0,所以CF=0
(21)shr:逻辑右移指令
;(1)将一个寄存器或内存单元中的数据向右移位; ;(2)将最后移出的一位写入CF中 ;(3)最高位用0补充 ;例如: mov al, 10000001b shr al, 1 ;结果:(al)=01000000b, CF=0 ;若移动位数大于1,则必须将移动位数放在cl中 mov al, 01010001b mov cl, 3 shr al, cl ;结果:(al)=00001010b, 因为最后移除的一位是0,所以CF=0
(22)int 常量:中断例程
总结:
由上述可以看出8086汇编有以下语法规律:
(1)指令格式有3种形式:
指令 目标 源
指令 目标
指令
(2)除int指令外,“目标”不能为常量
(3)当“目标”为段寄存器时,“ 源”只能为寄存器
(4)"目标“和"源"所占的内存字节大小应保证相同,
当双方显式知晓内存字节大小时,若不相同,则无法通过编译,
当仅有一方知晓内存字节大小时,数据处理则按该方内存字节大小计算,
当双方都不知晓内存字节大小时,应用byte/word/dword ptr显示声明,其中dword ptr仅用于一些特殊指令。
(5)转移指令有直接设置和位移偏移两种,其中位移偏移有距离限制。