汇编语言程序设计(三)之汇编程序
系列文章
汇编语言程序设计(一)
汇编语言程序设计(二)之寄存器
汇编程序
经过上述课程的学习,我们可以编写一个完整的程序了。这章开始我们将开始编写完整的汇编语言程序,用编译和连接程序将它们连接成可执行文件(如.exe文件)在DOS中运行。为了能够透彻地理解一个完整的程序(尽管它看上去十分简单),我们将经历一个漫长的过程。
1. 汇编程序的执行过程
下图描述了一个汇编程序从写出到最终执行的简要过程:
第一步:编写汇编源程序
使用文本编辑器(记事本、Notepad++等等),用汇编语言编写汇编源程序。这一步的工作是产生了一个存储源程序的文本文件
第二步:对源程序进行编译连接
使用汇编语言编译程序对源程序文件中的源程序进行编译,产生目标文件;再用连接程序对目标文件进行连接,生成可在操作系统直接运行的可执行文件。
可执行的文件包含两部分内容:
- 程序和数据:程序中的汇编指令翻译过来的机器码和源程序中定义的数据
- 相关的描述信息:程序多大,要占用多少内存空间等
三步:执行可执行文件中的程序
在操作系统中,执行可执行文件中的程序。此时,操作系统依照可执行文件中的描述信息,将可执行文件中的机器码和数据加载入内存,并进行相关的初始化(比如CS:IP指向第一条要执行的指令),然后由CPU执行程序。
2. 源程序
下面我们来看一下一段简单的程序:
assume cs:codesg
codesg segmentmov ax,0123Hmov bx,0456Hadd ax,bxadd ax,axmov ax,4C00Hint 21H
codesg ends
end
下面来对这个程序进行说明。在汇编语言程序中,包含两种指令,一种是汇编指令,一种是伪指令。汇编指令是有对应机器码的指令,可以被编译成机器指令,最终被CPU执行。而伪指令并没有对应的机器码,最终不会被CPU执行,而是会被编译器所执行。
- 我们先来看一下上述程序中出现的伪指令
- xxx segment
…
xxx ends
segment 和 ends 是成对使用的伪指令,这是在写可被编译器编译的汇编程序时必须要用到的一对伪指令。segment和ends的功能是定义一个段,segment说明一个段的开始,ends说明一个段的结束。在汇编源程序中,一个段必须要有一个名称标识,使用格式为:
段名 segment
…
段名 ends
像上述的程序中:(定义了一个代码段)
codesg segment ;定义了一个段,段的名称为codeseg,这个段从此开始;..
codesg ends ;名称为codeseg的段到此结束
当然一个源程序可以由多个段组成,可以用来存放代码、数据或者当做栈空间来使用。一个汇编源程序至少要有一个段:代码段。
2.end
end是一个汇编程序的结束标记,编译器在编译汇编程序的时候,如果碰到end,就会结束对源程序的编译。所以,我们在写程序的时候,程序结束的时候应该在结尾处加上伪指令end。
3.assume
assume:假设,它可以用来假设某一段寄存器和程序中的某一个用segment…ends定义的段相关联。在编程时,我们需要使用assume将有特定用途的段和相关的段寄存器关联起来。
例如上述程序中定义了一个名为codesg的代码段,我们需要将CS段寄存器和该段进行关联,格式则为:
assume cs:codesg
如果你要关联多个段:可以使用如下格式:
assume cs:codesg ds:datasg ss:stacksg
- 除了伪指令以外,还有最重要的汇编指令
编程的最终目的是让计算机完成一定的任务,而任务的完成由汇编指令来指示。汇编指令经过编译连接之后会转换成对应的机器码,如下图所示:
除了上述内容以外,我们还需要知道codesg的具体含义
它被称为标号。一个标号指代了一个地址。上述示例中codesg在segment前面,作为一个段的名称,最终在编译连接之后处理为一个段的段地址
认识上述的指令之后,我们再来看一下汇编程序的结构,从整体上来看一个汇编源程序应该需要什么
1.定义一个段(当然可以有多个段,但至少要有一个代码段
;比如 定义了一个段,段名为abc
abc segment
;..
abc ends
2.往段中加入汇编指令
abc segmentmov ax,2 ;这三条汇编语句的作用是求2^3add ax,axadd ax,ax
abc ends
3.指明程序在何处结束
abc segmentmov ax,2 ;这三条汇编语句的作用是求2^3add ax,axadd ax,ax
abc ends
end
4.指明代码段:人为设置代码段编译器并不会也这么认为
assume cs:abc
abc segmentmov ax,2 ;这三条汇编语句的作用是求2^3add ax,axadd ax,ax
abc ends
end
5.程序返回
上述讲解我们并没有讲述下面两条语句的作用,它们的作用是程序返回,类似于C语言中的return
mov ax,4C00H
int 21H
当然我们不用过多探究它们具体是什么意思,我们知道它们的功能是程序返回就可以了。所以上述程序还可以完善:(到此为止我们才算完成了一个较为完整的汇编源程序)
assume cs:abc
abc segmentmov ax,2 ;这三条汇编语句的作用是求2^3add ax,axadd ax,axmov ax,4c00Hint 21H
abc ends
end
源程序和程序的区别:我们称源程序文件中的所有内容成为源程序,比如1.asm中的内容。将源程序中最终由计算机执行、处理的指令或数据称为程序。
到此为止,我们已经接触了几个和结束相关的内容:段结束、程序结束、程序返回,下图展示了它们的区别:
3. 编译
上节内容已经可以完成一个源程序的编写,我们将其保存为1.asm之后就可以得到一个源程序文件。在这之后我们就可以对其进行编译了,下面我们来讲解编译过程。编译需要借助于相应的编译器,我们采用的masm汇编编译器。在上一章中我们已经讲解了如何进行源程序的编译连接过程,现在我们来详细看看编译的过程
进入DOS,并且进入到MASM目录(我们放在虚拟C盘中,如果不是的可以使用cd指令进入对应的目录)
如上图,运行masm后,首先显示出一些版本信息,然后提示输入将要被编译的程序文件的名(【.ASM】)。我们默认的扩展名是asm,如果要编译的源程序文件名为p1.asm,在这里只要输入p1即可。如果源程序文件名字不是asm的扩展名则需要输入全名。比如文件名为p1.txt,就要输入全名。当然我们输入文件名的目的是让系统根据对应的文件名找到对应的文件,我们输入文件名的时候还需要指明,这个文件的所在路径,比如文件p1.txt在c:code\下,则输入应为:c:code\p1.txt,如果你的可执行文件和masm文件在同一级目录,则可以直接输入名字即可,后续中间文件.obj同理。
使用masm编译了源文件之后会生成一个目标文件.obj,我们也可以制定这个文件的名字,示意如下:
上述的图片中默认生成1.OBJ,我们也可以设置为其他名字。后续还会有两个提示,分别是提示生成列表文件和引用文件,当然我们可以直接忽略这两个文件,他们属于中间文件,之后一直按下Enter键即可。
最终出现如下界面:
当我们完成了上述步骤之后就会出现一个新的文件1.obj。当然如果编译过程中出现错误的话将不会生成最终的目标文件。程序出错的标志:
-
程序中有Severe Errors
-
找不到所给出的源程序文件
4. 连接
在对源文件进行编译得到目标文件之后,我们需要对目标文件进行连接,从而得到可执行文件。在这里我们需要使用到连接工具,文件名为LINK.EXE。下面我们来看看连接的过程。
输入link之后,系统提示你输入要进行连接的目标文件(【.OBJ】),而Run File这一行则是提示你是否要指定生成的可执行文件的名字,不指定则为【】中默认的名字,上述图示中为指定默认可执行文件名为1.exe,之后一直按Enter即可。
当然上述图片中有一个警告错误:没有栈段。在这里我们不用理会它。这之后将生成一个可执行文件1.exe,后续直接在命令行输入1.exe即可执行该可执行文件
关于连接的作用:(仅作了解)
-
当源程序很大时,我们可以将它分为多个源程序进行编译,每个源程序都将产生一个目标文件,我们需要用连接程序将多个目标连接成一个可执行文件
-
程序中调用了某个库文件中的子程序,需要将这个库文件和该程序生成的目标文件连接在一起,最终生成一个可执行文件
-
一个源程序编译后,得到了存有机器码的目标文件,目标文件中的有些内容还不能直接用于生成可执行文件,需要经过连接程序将这些内容处理为最终的可执行信息。所以即使只有一个源文件和没有调用某个库的时候也需要进行连接
简化编译和连接的方法:
-
编译:masm 1.ASM;
-
连接:link 1.OBJ;
上述的两个指令制定了需要编译或者连接的文件,并且用;表示生成默认的文件,
编译指令中;表示默认生成1.OBJ
连接指令中;表示默认生成1.EXE
效果如下图
5. 程序执行过程的跟踪
我们可以利用Debug来跟踪一个程序的运行过程,这通常是必须要做的工作。我们写的程序在逻辑上不一定总是正确的,对于简单的错误,仔细检查一下源程序即可发现。但是对于隐藏较深的错误,就需要对程序的执行过程进行跟踪分析才可以发现了。
这个过程我们需要借助工具debug。debug可以将程序加载入内存,设置CS:IP指向程序的入口。
具体用法如下图所示:
接下来可以使用R命令来看一下各个寄存器的设置情况,如下:
可以看到此时的CS:IP指向076A:0000内存单元,也就是说第一条要执行的指令在该地址单元中。仔细看最后面:076A:0000 B80200 MOV AX,0002,这里指示了第一条指令的地址、机器码、对应的汇编的指令我们可以用u命令看一下其它的指令,如下图:
我们可以看到076A:0000~076A:000A都是程序的机器码
到这里我们可以来进行跟踪了,使用T命令进行单步执行,并观察每条指令执行后对寄存器的影响。
当然,到了int 21这条指令时我们需要用P命令执行,如下图:
上图执行完INT 21之后显示出 “Program terminated normally”,返回到debug中。表示程序正常结束
当然,需要注意的是在执行INT 21这条语句时需要用P命令去执行。
练习:在这里大家可以通过调试一个汇编程序,在每步执行后,相关寄存器内容的改变。程序如下:
相关文章:

汇编语言程序设计(三)之汇编程序
系列文章 汇编语言程序设计(一) 汇编语言程序设计(二)之寄存器 汇编程序 经过上述课程的学习,我们可以编写一个完整的程序了。这章开始我们将开始编写完整的汇编语言程序,用编译和连接程序将它们连接成可…...

用二极管和电容过滤电源波动,实现简单的稳压 - 小水泵升压改装方案
简而言之,就是类似采样保持电路,当电源电压因为电机启动而骤降时,用二极管避免电容电压跟着降低,从而让电容上连接的低功耗芯片有一个比较稳定的供电电压。没什么特别的用处,省个LDO 吧,电压跌幅太大的时候…...

【数据结构与算法】数据结构有哪些?算法有哪些?
1. 算法与数据结构总览图 2.常用的数据结构 2.1.数组(Array) 数组是一种聚合数据类型,它是将具有相同类型的若干变量有序地组织在一起的集合。数组可以说是最基本的数据结构,在各种编程语言中都有对应。一个数组可以分解为多个数…...

使用Element-UI展示数据(动态查询)
学习内容来源:视频P4 本篇文章进度接着之前的文章进行续写 精简前后端分离项目搭建 Vue基础容器使用 目录选择组件修改表格组件修改分页组件增加后端接口前端请求数据接口页面初始化请求数据点击页码请求数据选择组件 在官方文档中选择现成的组件,放在页…...
lamda 表达式例子全集
1、List 转 map 1.1、key(Model属性) value Model Map<String, Model> modeMap List.stream().collect(Collectors.toMap(Model1::属性get方法, v -> v, (p1, p2) -> p1)); 1.2、key(Model1属性) value Model2 Map<String, Model1> model2Map List.stream…...

计算机网络第八版——第一章课后题答案(超详细)
第一章 该答案为博主在网络上整理,排版不易,希望大家多多点赞支持。后续将会持续更新(可以给博主点个关注~ 【1-01】计算机网络可以向用户提供哪些服务? 解答:这道题没有现成的标准答案,因为可以从不同的…...

嵌入式和Python(二):python初识及其基本使用规则
目录 一,python基本特点 二,python使用说明 ● 两种编程方式 ① 交互式编程 ② 脚本式编程 ● python中文编码 ● python行和缩进 ● python引号 ● python空行 ● python等待用户输入 ① 没有转换变量类型 ② 转换变量类型 ● python变…...

C语言详解双向链表的基本操作
目录 双链表的定义与接口函数 定义双链表 接口函数 详解接口函数的实现 创建新节点(BuyLTNode) 初始化双链表(ListInit) 双向链表打印(ListPrint) 双链表查找(ListFind) 双链…...

面试必须要知道的常见排序算法
以下排序均为升序 1.直接插入排序 具体思想 把待排序的数据按大小比较插入到一个已经排序好的有序序列中,直到所有的待排序数据全部插入到有序序列中为止.实际生活中,我们平常斗地主摸牌时,就用到了插入排序的思想. 当插入第n个数据时,前面n-1个数据已经有序;第n个数据依次与前…...

Kubernetes之服务发布
学了服务发现后,svc的IP只能被集群内部主机及pod才可以访问,要想集群外的主机也可以访问svc,就需要利用到服务发布。 NodePort Nodeport服务是外部访问服务的最基本方式。当我们创建一个服务的时候,把服务的端口映射到kubernete…...
【第二章】谭浩强C语言课后习题答案
1. 什么是算法?试从日常生活中找3个例子,描述它们的算法 算法:简而言之就是求解问题的步骤,对特定问题求解步骤的一种描述。 比如生活中的例子: 考大学首先填报志愿表、交报名费、拿到准考证、按时参加考试、收到录取通知书、按照日期到指定学校报到。 去北京听演唱会首先…...
PostgreSQL和PostGISWGS84和CGCS2000与GCJ02和BD09坐标系与之间互转
– 如果转换后结果为null,查看geom的srid是否为4326或者4490 WGS84转GCJ02 select geoc_wgs84togcj02(geom) from test_table GCJ02转WGS84 select geoc_gcj02towgs84(geom) from test_table WGS84转BD09 select geoc_wgs84tobd09(geom) from test_table BD09转WGS84 select …...

数据结构——链表讲解(2)
作者:几冬雪来 时间:2023年3月5日 内容:数据结构链表讲解 目录 前言: 剩余的链表应用: 1.查找: 2.改写数据: 3.在pos之前插入数据: 4.pos位置删除: 5.在pos的后…...

Elasticsearch:图片相似度搜索的 5 个技术组成部分
作者:Radovan Ondas,Bernhard Suhm 在本系列博文的第一部分中,我们介绍了图像相似度搜索,并回顾了一种可以降低复杂性并便于实施的高级架构。 此博客解释了实现图像相似性搜索应用程序所需的每个组件的基本概念和技术注意事项。 学…...

【CVPR2022】Class Re-Activation Maps for Weakly-Supervised Semantic Segmentation
论文标题:Class Re-Activation Maps for Weakly-Supervised Semantic Segmentation收录:CVPR 2022paper: https://arxiv.org/abs/2203.00962code: https://github.com/zhaozhengChen/ReCAM解读:https://zhuanlan.zhihu.com/p/478133151https:…...

PMP项目管理项目运行环境
目录1 概述2 事业环境因素和组织过程资产3 组织系统3.1 概述3.2 组织治理框架3.2.1 治理框架3.2.2 项目治理3.3 管理要素3.4 组织结构类型3.4.1 组织结构类型3.4.2 项目管理办公室1 概述 项目所处的环境可能对项目的开展产生有利或不利的影响,这些影响的两大主要来…...
Vue 3.0 渲染函数 【Vue3 从零开始】
Vue 推荐在绝大多数情况下使用模板来创建你的 HTML。然而在一些场景中,你真的需要 JavaScript 的完全编程的能力。这时你可以用渲染函数,它比模板更接近编译器。 让我们深入一个简单的例子,这个例子里 render 函数很实用。假设我们要生成一些…...

西电软件体系结构核心考点汇总(期末真题+核心考点)
文章目录前言一、历年真题二、核心考点汇总2.1 什么是软件体系架构?(软件体系结构的定义)2.2 架构风格优缺点2.3 质量属性2.4 质量评估前言 主要针对西安电子科技大学《软件体系结构》的核心考点进行汇总。 【期末期间总结资料如下】 针对西电计科院软件工程专业大三《软件体…...

SRS源码分析-SDP内容解析
前言 在学习SRS的RTC模块之前,首先来分析下SRS在将rtmp推流转成rtc流,通过浏览器拉取webrtc流场景下产生的SDP内容 SDP格式介绍 SDP数据是文本格式,由多个 <key><value> 表达式构成,<key>的值只能是一个字符…...
HTML 颜色
HTML 颜色 HTML 颜色采用的是 RGB 颜色,是通过对红 (R)、绿 (G)、蓝 (B) 三个颜色通道的变化以及它们相互之间的叠加来得到各式各样的颜色的,RGB 即是代表红、绿、蓝三个通道的颜色。 Color Values HTML 颜色由一个十六进制符号来定义,这个符…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...

376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...

跨链模式:多链互操作架构与性能扩展方案
跨链模式:多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈:模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展(H2Cross架构): 适配层…...
Spring AI 入门:Java 开发者的生成式 AI 实践之路
一、Spring AI 简介 在人工智能技术快速迭代的今天,Spring AI 作为 Spring 生态系统的新生力量,正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务(如 OpenAI、Anthropic)的无缝对接&…...

12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...
Spring AI与Spring Modulith核心技术解析
Spring AI核心架构解析 Spring AI(https://spring.io/projects/spring-ai)作为Spring生态中的AI集成框架,其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似,但特别为多语…...
安卓基础(aar)
重新设置java21的环境,临时设置 $env:JAVA_HOME "D:\Android Studio\jbr" 查看当前环境变量 JAVA_HOME 的值 echo $env:JAVA_HOME 构建ARR文件 ./gradlew :private-lib:assembleRelease 目录是这样的: MyApp/ ├── app/ …...
AGain DB和倍数增益的关系
我在设置一款索尼CMOS芯片时,Again增益0db变化为6DB,画面的变化只有2倍DN的增益,比如10变为20。 这与dB和线性增益的关系以及传感器处理流程有关。以下是具体原因分析: 1. dB与线性增益的换算关系 6dB对应的理论线性增益应为&…...
SQL慢可能是触发了ring buffer
简介 最近在进行 postgresql 性能排查的时候,发现 PG 在某一个时间并行执行的 SQL 变得特别慢。最后通过监控监观察到并行发起得时间 buffers_alloc 就急速上升,且低水位伴随在整个慢 SQL,一直是 buferIO 的等待事件,此时也没有其他会话的争抢。SQL 虽然不是高效 SQL ,但…...

力扣热题100 k个一组反转链表题解
题目: 代码: func reverseKGroup(head *ListNode, k int) *ListNode {cur : headfor i : 0; i < k; i {if cur nil {return head}cur cur.Next}newHead : reverse(head, cur)head.Next reverseKGroup(cur, k)return newHead }func reverse(start, end *ListNode) *ListN…...