80x86汇编—汇编程序基本框架

news2025/1/19 2:55:52

文章目录

  • First Program
  • 指令系统
  • 伪指令
    • 数值表达式
  • 程序框架解释
  • int 21 中断

通过一个基本框架解释各个指令和用处,方便复习。所以我认为最好的学习顺序就是先看一段完整的汇编代码程序,然后给你逐个逐个的解释每一个代码是干嘛用的。然后剩下的还有很多指令或者伪指令会在以后用到的时候再根据具体上下文使用,我认为汇编就是实战出来的,很多指令根本不用记忆,需求出来了,你用了后就知道是怎个事了。

First Program

经典Hello Word!
代码如下:

stack segment
	db 10 dup(00)
	topSP equ this word ;栈顶指针地址标号
stack ends
datas segment
	string db 'Hello Word!', 0Dh, 0AH, '$'
	db 10 dup (?)
datas ends
code segment
assume cs:code,ss:stack,ds:datas  
start:
	mov ax, datas
	mov ds, ax
	mov dx, offset string
	mov ah, 09h
	int 21h	
	mov ax, 4c00h
	int 21h
code ends
end start


编译->链接->运行,经过三个步骤就能够使用debug对生成的exe文件调试了
编译命令:masm 名.asm;
链接命令:link 名.obj; (obj文件是在masm后自动生成的)
运行命令:名.exe(exe文件是在link后自动生成的)
debug调试exe命令:debug 名.exe

在这里插入图片描述

指令系统

点击复习指令系统

伪指令

伪指令都不占程序空间,因为在编译过程中,你的伪指令全部帮你翻译好了,然后变成机器指令后就只能按照机器指令, 机器指令没有伪指令,伪指令是方便人类阅读代码的,可以通过汇编编译器翻译对应的计算机能够执行的机器指令,比如:fun我把他作为一个函数入口地址,我们编译器翻译的时候不会再fun这一处存fun的地址空间,而是假设你有用到fun这个函数入口地址的话,在你使用该标号的位置处将该标号翻译成fun的入口地址就是fun的下一条指令。

  • 标号(他不是伪指令,是一个不占空间但又不是伪指令的东西)
    可以理解为变量名,变量的名字可以随便起,这个和下面介绍的名字指令差不多。
    标号后一定要接了一个冒号:,表示这个标号可以作为一个地址,地址和下一个指令地址一样。(虽然不是伪指令,但是他也不占程序空间,用来标记程序中某些位置的)

    随便名标号:
    mov ax ,bx
    loop 随便名标号
    ...
    
  • 名字(就是程序中Stack 或者 datas这些,可以随便命名)
    可以理解为变量名,用来记录当前位置的下一条指令或者数据的地址,所以利用这个特点我们就可以用它来定位是一件十分轻松的事情,因为他不占空间地址,写一条指令上去后就可以直接用这个指令的名字来表示地址了,很方便。(这里我程序里面也用到了我给栈顶指针起了个名字topSP ,最显著的特点就是用来当栈顶指针,妈妈再也不用担心我计算指针位置错误了!又因为他不占空间,有了这个标志就直接用就可以找到栈顶指针,不会影响后面的代码位置关系)

    随便名字段 segment 
    随便名字段 ends
    随便名字段 db 'hello'
    ...
    
  • ASSUME
    翻译就是:假设、设想
    在汇编中是指明段寄存器的段名,格式为:assume 段寄存器:随便名字段
    随便名字段是段地址,是段地址,是段地址!!
    assume可以放在开头也行,不一样放在数据段定义的后面,因为assume是伪指令,汇编编译过程中会帮你翻译好。
    很明显这个是给自己编写的段一个归属,比如我datas段就是专门给ds段的,所以在assume中就是ds:datas,这就将我们的datas段数据连接到了数据段了,说到连接,那其实assume的本质就是为了汇编在link的时候将自己写的段设置到正确的段位置。
    注意事项:assume是 给自己定义的数据块指明回家的路,回到自己的段内,意思是DS冒号后面你的数据段属于DS段,那你要是用你他的时候就需要在DS里面访问(这里是一个坑,下面我接着说很重要的一点)
    还有一点很重要:我们assume定义完成后希望在code代码里面使用这段数据的话,其实代码里面还需要自己将DS段设置为随便名字段对应的段地址,我们的随便名字段就是段地址。
    因此正确使用我们定义的数据段因该是:

    assume cs:随便名字段,ds:datas, ss:随便名字段3
    mov ax, datas
    mov ds, ax 
    

    将datas段地址给到我们的ds段,由于前面设置了ds:datas,所以编译器看得懂你这段datas数据,否则定义在其他段了你又在ds使用的话就会出问题。datas可以随便起名字

  • OFFSET 标号或者名字 / SEG 标号或者名字
    offset是返回标号在他自己的段的偏移地址
    seg是返回标号所在的段地址
    一般用法:

    mov ax, offset string 
    mov ax, seg string;这里返回的是string所在的段地址
    

    通过验证也可以知道seg是真的返回了string的段地址,所以有时候不知道string的段地址的名字叫啥比如忘记了datas的话,我们就可以通过这种方式获取段地址,如下图所示
    在这里插入图片描述

  • DUP
    格式:重复次数 DUP (要重复的东西)
    DUP一般是用来赋初值的
    要重复的东西:这里可以是问号,可以是数值,也可以是字符串等等,字符串也给你重复好多次。
    ?问号表示不确定是什么,如下图,通过验证后,8086debug中显示是0的状态。
    在这里插入图片描述

    db 10 dup(00)
    db 10 dup (?)
    
  • DB/DD/DW
    DB字节 8字节
    DD单字 16字节
    DW双字 32字节
    用处就是当你定义一个数据段的时候可以指明一个元素占多少字节
    好比上一个DUP中用到的就是DB,表示他每一个重复的元素都是占8个字节大小

    db 10 dup(00)
    
  • $
    有时候表示当前地址
    比如:使用 $ 和相对偏移计算mov bx, $ - 标号
    有时候表示字符串结束符,当我们int 21中 09h的功能就是当遇到$字符的时候停止输出字符。
    比如:

    前面我们定义了string db 'Hello Word!', 0Dh, 0AH, '$'
    所以int 2109号中断程序会在打印字符中遇到$的时候停止
    mov ah, 09h
    nt 21h	
    
  • EQU / =
    equal相等,=也是相等
    格式:

    随便 equ 12
    随便2 = 13
    

    那以后代码中出现 了随便,就会在汇编编译过程汇总将随便这个名字换成12同理随便2换成13

  • PTR
    在指令系统讲过了,就是指明类型的意思
    下面介绍几个用法即可
    在这里插入图片描述

  • THIS
    用法:THIS 类型名
    这种用法搭配equ和等号简直无敌,因为是伪指令所有使用他的时候不占程序空间内存,单单用作给编译器翻译,让编译器去做而已,翻译机器码后是没有这个语句的
    让我感到优雅的一个语句就是:topSP equ this word ;栈顶指针地址标号
    基本一看就懂了,最牛逼的是opSP equ this word伪指令他不占内存。翻译的时候他名字对应的地址是等于下一个指令的地址,并且不会影响你该语句后面的数据段或者代码段位置。我们以后想用栈的时候直接使用topSP这个名字就行,真的是一箭双雕,太优雅了!

    stack segment
    	db 10 dup(00)
    	topSP equ this word ;栈顶指针地址标号
    stack ends
    

    下图也可以看出确实topSP没有他的内存空间
    在这里插入图片描述
    还有一种老师的用法,就是通过地址相同,因为不占空间,但是地址是下一条指令地址,我们直接将f_jump 相等equ与far,因为不指明跳转范围的标号都是段内转移,但是由于f_jump等于下一个指令的地址,但是他的属性类型是far,所以只要用他的名字就可以实现段间转移(确实有点反人类指令)
    在这里插入图片描述

  • TYPE
    这个就是typeofC语言中的函数一样,
    type 标号或者名字,能够返回该标号或者名字的类型大小,以字节作为单位,
    比如一个标号s是段内转移,那他就是一个byte,返回的就是1,假设像我们f_jump的话,我们type一下他就返回2,因为far是段间

mov ax, type f_jump ;这里等于 mov ax, 2
mov ax, type n_jump ;这里等于 mov ax, 1
  • SIZEOF
    这个指令就是量整个变量所占的空间,可以理解为量我们的数据段首地址,然后就会返回你这个数据段的占的字节大小。

  • LENGTHOF
    这个就是量你元素个数,数据段中我们可能有多个相同大小的多个元素,那么这个指令就是用来量长度的,量个数的。

  • SIZEOF = TYPE × LENGTHOF

数值表达式

这个就有点推翻以前说的格式了,我们以前的指令格式都是:
操作指令 目的操作数,源操作数
但当你使用数值表达式的话就不一样了,就跟我们平常的写式子的顺序是一样的。

mov ax,3*4+5 ;等价于 mov ax,17

目前看起来很正常,因为我们的乘除法这样使用的在高级语言中也是可以。

or al,03h AND 45h ;等价于 or al,01h

这种就有点迷惑了,首先我们学的and指令都是:and 操作数1,操作数2
但是这种就是跟你写数学式子一样
也是数值表达式的一种特色吧,直接用and没给你改成其他的指令,直接拿硬指令,所以通过这个式子例子就可以猜到后面的用法了,
也就是说当我们需要一种计算的式子的话,可以放到源操作数中进行,不用手动计算,工作交给编译器,编译完成后会直接形成结果,所以不会影响执行速度。

mov al,0101b SHL (2*2);等价于 mov al,01010000b

这个指令就真的是推翻了我在指令系统里面的解释,指令系统中提到过shl如果要移位超过1位的话就要将移位个数存进cl中,然后shl ax, cl才可以多次移位,但是在数值表达式中就直接推翻了,但是 前提是你需要按照数值表达式的格式来写才可以,所以我们可以0101b SHL (2*2),但是这种形式是一个结果,所以我们一般都是需要放在源操作数,然然后送进目的操作数中。

程序框架解释

最后会在int中断号给出相关示例代码,这里只给出模版。

  • assume设置自己定义的数据段时属于哪一个段寄存器的,一般都会有assume,因为我们一般情况下是会定义一个代码段,assume一般是放在开头第一段或者code segment下一行。

  • codes segment …
    codes ends

    代码段,段都是用segment,
    开头段都需要 [段名 segment]
    结束段都需要 [段名 ends]

  • start:…
    end start

    start是程序入口,需要在程序最后结尾添上end start
    start是可以随便起名字

  • 代码段和入口程序标号通常是交织在一起的,我们就按照正常的格式写。如下:

codes segment
       start:
       ......
       ......
       ......
codes ends
end start

整合一下整个框架就是:
(记住,要用数据段的时候千万不要漏掉将ds或者其他段用对应的标号修改,比如:mov ax, datas mov ds, ax,修改ds段地址)

assume cs:codes, ss:stacks, ds:datas
stacks segment
... ... ...
stacks ends
datas segment
... ... ...
datas ends
codes segment
	start:
	... ...
	... ...
	... ...
	... ...
codes ends
end start 

int 21 中断

这里是最后一个知识点,因为想要入门一个简单的程序需要用到,但是不会讲的很详细,int中断本来就是一个很大的章节需要慢慢学习,很多功能都能解决很多问题。这里就简单介绍几个常用的功能号

我们在程序中经常会看到这么一段

mov ah, 09h
int 21h	
mov ax, 4c00h
int 21h

int 21h是一个DOS系统中断,21h里面包含很多功能,但是需要用寄存器来空间。
怎么控制?
最简单的一个控制,就是当ax等于09h的时候就是打印一个ds段里面的dx偏移地址的字符串,遇到$符号就停止,所以这里终于解释明白了我们First程序的字符串是怎么打印出来的了。
因此可以想象一下,我们int 21h中断有很多功能,那我们不仅仅是一个打印功能,还有很多,有可能是al来控制,也有可能是ah来控制,也有可能是其他寄存器,当然我不知有没有其他寄存器i,只是告诉使用int 21h功能的时候,记得设置功能调用号和参数。
09功能号,参数是ds:dx作为字符串入口
所以就是打印出来一个Hello Word!
4c00h就是意味着代码段程序正常退出
但是不是整个程序结束,整个程序结束时end start,当然 start是随便名,在你代码段开始的start:自己定义名字的。

下面介绍的比如09h数字是ax或者al/ah对应的数字所对应21h中断的功能程序是什么。当然09h上面说过了是打印ds:dx首地址的字符串,遇到$停止
注意:下面需要将中断号传到ah而不是ax,除了4c00需要传到ax,因为他是16位。

  • 09h
    功能:在显示器输出指定的字符串
    DS:DX=欲显示字符串在主存中的首地址
    字符串应以$(24H)结束
    可以输出回车(0DH)和换行(0AH)字符产生回车和换行的作用
    在这里插入图片描述
    在这里插入图片描述
    示例代码如下:

    ;使用09中断号打印出来Hello Word!
    assume cs:code, ds:datas
    
    datas segment 
    	string db 'Hello Word!', 0dh, 0ah, '$'
    datas ends
    
    code segment
    	start:
    		mov ax, datas 
    		mov ds, ax ;设置datas对应的数据段
    		mov dx, offset string ;这里一定要用offset,
    		mov ah, 09h ;打印ds:dx字符串,遇到$停止
    		int 21h
    		mov ax, 4c00h ;代码段结束
    		int 21h
    code ends
    end start 
    
  • 01h
    功能:获得按键的ASCII代码值
    功能号:AH=01H
    出口参数:AL=字符的ASCII码
    调用此功能时,若无键按下,则会一直等待,直到按键后才读取该键值
    在这里插入图片描述
    示例代码如下:

    ;使用01h中断号,循环录入Hello Word!  也就是说重复录入11个字符
    assume cs:codes, ds:datas
    
    datas segment
    	string db 11 dup (0), 0dh, 0ah, '$'
    	endString equ this byte
    datas ends
    
    codes segment
    	start:
    	mov ax, datas
    	mov ds, ax
    	mov cx, 11
    	mov al, 0
    	mov ah, 01h
    	mov si, 0
    	mov bx, offset string 
    	
    	s:
    	int 21h
    	mov [bx][si], al 
    	inc si
    	loop s
    	
    	mov dx, offset string 
    	mov ah, 09h
    	int 21h
    	
    	mov ax, 4c00h
    	int 21h
    	
    codes ends
    end start 
    
    
  • 02h
    功能:在显示器当前光标位置显示给定的字符,光标右移一个字符位置。如按Ctrl-Break或Ctrl-C则退出
    功能号:AH=02H
    入口参数:DL=字符的ASCII码
    既然他是显示一个字符就右移一个字符,那就是说我们可以利用循环每调用一次显示一个字符也不会覆盖住我们先前显示的字符了。理论存在,开始实现!
    在这里插入图片描述
    示例代码如下:

assume cs:codes, ds:datas

datas segment
	string  db 'Hello Word!'
			beginString equ this byte
datas ends

codes segment
	start:
	mov dx, seg string
	mov ds, dx
	
	mov ah, 02h
	mov cx, offset beginString - offset string
	mov si, 0
	s:
	mov dl, [offset string][si]
	inc si
	int 21h
	loop s
	
	mov ax, 4c00h
	int 21h
	
codes ends
end start 

  • 0ah
    功能:执行该功能调用时,用户按键,最后用回车确认
    功能号:AH=0AH
    入口参数:DS:DX=缓冲区首地址
    本调用可执行全部标准键盘编辑命令;用户按回车键结束输入,如按Ctrl+Break或Ctrl+C则中止

    细节:首先这个其实是有三个参数,
    就拿datas段来举例子

    datas segment
    string  db 12 ;这个是用来表示你最大可以输
    			  ;入多少个字符,这里是包含回车的,不要忘记
    			  
    		db 0 
    		;这里是表示你的输入字符个数,
    		;会在你输入的时候自动计算,然后一般是0
    		
    		db 12 dup(0) 
    		;这里是存你输入的字符串,
    		;所以你要输出的话就要记得将string 地址偏移地址
    		;偏移到该为止,这里由于是db大小所以偏移两个byte。
    

    下图证明了无论你第二个参数设置什么,都是默认帮你计算好最终你输入多少个字符的,我这里是输入了10个字符,但是我设置的是11,最后结果是a=10那就代表是强制性计数器。
    在这里插入图片描述
    下面是执行结果,输入字符串直到回车结束,但是我们这里设置了12个加上回车符号那就是只能够输入11个字符,所以刚好输入Hello Word!,然后再利用09h中断号再次找到字符串入口输出。
    在这里插入图片描述
    下面是示例代码:

    assume cs:codes, ds:datas
    
    datas segment
    	string  db 12
    			db 11
    			beginString equ this byte
    			db 12 dup (0)
    			db 0dh, 0ah, '$'
    datas ends
    
    codes segment
    	start:
    	mov dx, seg string
    	mov ds, dx
    	mov dx, offset string 
    	mov ah, 0ah
    	int 21h
    	
    	mov ah, 09h
    	mov dx, offset beginString
    	int 21h
    	
    	
    	mov ax, 4c00h
    	int 21h
    	
    codes ends
    end start 
    
    
  • 0bh
    功能:仅判断当前是否有按下的键,设置AL后退出
    功能号:AH=0BH
    出口参数:AL=0,当前没有按键;
    AL=FFH,当前已经按键。
    注意的是这里讲的是出口参数,是用来判断你该是否按下按键了

    这个就不打算写示例了,因为就是用来测试有没有按键事件发生的一个中断号,可能在写游戏程序用到很多。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1331128.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

linux的主线程提前子线程退出以及线程分离

主线程提前退出 如果主线程没有等待子线程提前退出,可能会发生以下情况: 子线程继续运行:如果主线程退出,但子线程仍在执行任务,子线程将继续独立运行。子线程的生命周期不受主线程控制,直到子线程自行完成…

Latex生成的PDF中加入书签/Navigation/导航

本文参考:【Latex学习】在生成pdf中加入书签/目录/提纲_latex 书签-CSDN博客 (这篇文章写的真的太棒了!非常推荐) 题外话,我的碎碎念,这也是我如何提高搜索能力的办法:想在Latex生成的PDF中加入…

python脚本 链接到ssh服务器 快速登录ssh服务器 ssh登录

此文分享一个python脚本,用于管理和快速链接到ssh服务器。 效果演示 🔥完整演示效果 👇第一步,显然,我们需要选择功能 👇第二步,确认 or 选择ssh服务器,根据配置文件中提供的ssh信息,有以下情况 👇场景一,只有一个候选ssh服务器,则脚本会提示用户是否确认链…

【hcie-cloud】【9】华为云Stack_Deploy部署工具介绍

文章目录 前言华为云Stack Deploy简介华为云Stack Deploy工具简介华为云Stack Deploy工具部署范围华为云Stack Deploy工具节点网络要求华为云Stack Deploy工具部署流程 华为云Stack Deploy功能介绍部署工具工程场景部署流程介绍创建工程 - 基本信息填写创建工程 - 基本参数选择…

【ITK库学习】使用itk库进行图像配准:“Hello World”配准

目录 1、itkImageRegistrationMethod / itkImageRegistrationMethodv42、itkTranslationTransform3、itkMeanSquaresImageToImageMetric / itkMeanSquaresImageToImageMetric44、itkRegularStepGradientDescentOptimizerv / itkRegularStepGradientDescentOptimizerv4 图像配准…

0基础学习VR全景平台篇第130篇:曝光三要素—感光度

上课!全体起立~ 大家好,欢迎观看蛙色官方系列全景摄影课程! 众所周知,摄影是一门用光的艺术。随着天气、地点、时间的变化,我们所处环境的光线也随之发生改变。而在不同的环境下该如何去正确的调节曝光,是…

Spring security之授权

前言 本篇为大家带来Spring security的授权,首先要理解一些概念,有关于:权限、角色、安全上下文、访问控制表达式、方法级安全性、访问决策管理器 一.授权的基本介绍 Spring Security 中的授权分为两种类型: 基于角色的授权&…

【Angular】Angular中的最差实践

自我介绍 做一个简单介绍,酒架年近48 ,有20多年IT工作经历,目前在一家500强做企业架构.因为工作需要,另外也因为兴趣涉猎比较广,为了自己学习建立了三个博客,分别是【全球IT瞭望】,【…

4.4【共享源】克隆实战开发之截屏(二)

三,显示器截图 screen_read_display() 函数则用于捕获显示器的屏幕截图。我们需要在特权上下文中工作,以便可以完全访问系统的显示属性。我们可以通过调用具有 SCREEN_DISPLAY_MANAGER_CONTEXT 上下文类型的 screen_create_context() 来创建特权上下文。进程必须具有 root 的…

使用minio实现大文件断点续传

部署 minio 拉取镜像 docker pull minio/minio docker images新建映射目录 新建下面图片里的俩个目录 data(存放对象-实际的数据) config 存放配置开放对应端口 我使用的是腾讯服务器所以 在腾讯的安全页面开启 9000,9090 两个端口就可以了(根据大家实际…

使用python对windows/win11进行属性设置

有一个个人的需求,针对windows系统进行属性设置,这里以对鼠标的左右键主键进行切换为例,进行了研究,以当前win11系统为基础进行了更动。 首先是对于如果打开windows系统下的鼠标设置,有以下几种办法: 添加…

十大经典排序算法(个人总结C语言版)

文章目录 一、前言二、对比1.排序算法相关概念1.1 时间复杂度1.2 空间复杂度1.3 排序方式1.4 稳定度 2.表格比较3.算法推荐3.1 小规模数据3.2 中等规模数据3.3 大规模数据3.4 特殊需求 三、排序算法1.冒泡排序(Bubble Sort)1.1 简介1.2 示例代码&#xf…

使用互斥锁(Mutex)管理共享资源

在Go中确保并发安全性 并发是Go中的一个强大功能,它允许多个Goroutines(并发线程)同时执行。然而,伴随着强大的功能也带来了大量的责任。当多个Goroutines并发地访问和修改共享资源时,可能会导致数据损坏、数据竞争&a…

网络爬虫之Ajax动态数据采集

动态数据采集 规则 有时候我们在用 requests 抓取页面的时候,得到的结果可能和在浏览器中看到的不一样,在浏览器中可以看到正常显示的页面教据,但是使用 requests 得到的结果并没有,这是因为requests 获取的都是原始的 HTML 文档…

【小白攻略】php 小数转为百分比,保留两位小数的函数

php 小数转为百分比 首先,最简单直观的方法是利用PHP内置的number_format函数。该函数可以对一个数字进行格式化,并可以设置小数点后的精度。通过将小数乘以100,再用number_format函数将结果格式化为百分比形式,即可达到将小数转为…

uniapp怎么动态渲染导航栏的title?

直接在接口请求里面写入以下: 自己要什么参数就写什么参数 本人仅供参考: this.name res.data.data[i].name; console.log(名字, res.data.data[i].name); uni.setNavigationBarTitle({title: this.name}) 效果:

图论 | 网络流的基本概念

文章目录 流网路残留网络增广路径割最大流最小割定理最大流Edmonds-Karp 算法算法步骤程序代码时间复杂度 流网路 流网络: G ( V , E ) G (V, E) G(V,E) 有向图,不考虑反向边s:源点t:汇点 c ( u , v ) c(u, v) c(u,v)&#xff…

CSS:浮动

CSS:浮动 浮动效果浮动方式 float浮动特性标准流脱标脱标的影响脱标的影响范围 清除浮动清除浮动原理 clear基于clear的清除浮动方式额外标签法:afert伪元素法双伪元素法 清除浮动原理 BFCBFC定义BFC布局规则创建一个BFC基于BFC的清除浮动方式父级添加overflow法 浮…

物理模拟重力 斜抛运动计算 抛物线计算

物理模拟重力 斜抛运动计算 抛物线计算 一、介绍二、原理三、实现如下PhysicsUtil.cs 工具类Missile.cs 四、资源分享 一、介绍 模拟Unity原始重力系统进行重写,可是实现发射到指定目标位置并能继续当前力进行自身的弹力与摩擦继续运动 二、原理 将Unity原始不受控…

鸿蒙开发4.0-ArkTS与H5的交互

ArkTS侧与H5的交互 首先在开发H5页面(输入框和金额选择部分)前需要实现JSBridge桥接打通两侧的交互。开发者可以在ArkTS侧定义一个JSBridge类,在类中封装call方法以及initJsBridge方法。 准备异步执行脚本,在脚本中声明一个JSBri…