经过学习。对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)转移指令有直接设置和位移偏移两种。当中位移偏移有距离限制。