热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

汇编语言(王爽)第六章检测点与实验5

检测点6.1(1)依次用内存0:0~15单元中的内容改写程序中的数据,补全程序:assumecs:codesgcodesegmentdw0123h,0456h,0789h,0ab

检测点6.1

(1)依次用内存0:0~15单元中的内容改写程序中的数据,补全程序:

assume cs:codesg
code segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h

start:
mov ax,0
mov ds,ax
mov bx,0

mov cx,8
s: mov ax,[bx]

mov cs:[bx],ax ;确定目标区域段地址和偏移地址
add bx,2
loop s

mov ax,4c00h
int 21h

codesg ends
end start

(2)程序实现依次用0:0~15单元的内容改写程序中数据,数据传送用栈来进行。栈空间设置在程序内,补全程序:

assume cs:codesg
code segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
dw
0,0,0,0,0,0,0,0,0,0 ;10个字空间作为栈

start:
mov ax,cs ;设置栈的段地址
mov ss,ax
mov sp,1ah ;cs:0~cs:f为原始数据空间,cs:10~cs:19为栈空间,初始为空,栈顶指向下一单元

mov ax,0
mov ds,ax
mov bx,0
mov cx,8

s: push [bx] ;先把0:0处的字单元内容入栈
pop cs:[bx] ;再把栈顶内容出栈放入程序数据段中
add bx,2
loop s

mov ax,4c00h
int 21h

codesg ends
end start

实验5 编写、调试具有多个段的程序

<程序加载后,ds:0~ff为PSP区域,(ds+10H):0为整个程序的入口,如程序依序设置有data\stack\code3个数据段区域,其中设data和stack段各自为16个字节,那么程序加载后(还未运行前):(ds+10h)则为data的入口段地址;(ds+10h+1)为stack的入口段地址;(ds+10h+2)为code的入口段地址;>

1、编译连接下面程序,用debug加载、跟踪,回答问题。

assume cs:codesg,ds:data,ss:stack
data segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
data ends

stack segment
dw
0,0,0,0,0,0,0,0
stack ends

code segment
start:
mov ax,stack
mov ss,ax
mov sp,16 ;将逻辑上设置的栈段真正设为栈段区域

mov ax,data ;设置数据区域段地址
mov ds,ax

push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]

mov ax,4c00h
int 21h

code ends
end start

程序先把数据段前两个依序入栈,再出栈返回到各数据原先位置。

(1)CPU执行程序,程序返回前,data段中的数据为多少?

执行程序后,data段中的数据不变,为原始数据。

(2)CPU执行程序,程序返回前,cs=13feh,ss=13fdh,ds=13fch.

(3)程序加载后,code段地址设为X,则data段地址为(x-2),stack段的段地址为(X-1).

注:这里提问的是data段和stack段的段地址分别是多少,程序加载时候,还没有运行,此时的data段只是逻辑上的数据区,并还没有设置ds指向;stack也如此,只是设想里的栈区。只有程序运行后才成为真正的ds指向数据区和栈段。通过(2)也可以验证(3)。

 

2、编译连接下面程序,用debug加载、跟踪,回答问题。

assume cs:code,ds:data,ss:stack
data segment
dw 0123h,0456h
data ends

stack segment
dw
0,0
stack ends

code segment
start:
mov ax,stack
mov ss,ax
mov sp,16

mov ax,data
mov ds,ax

push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]

mov ax,4c00h
int 21h

code ends
end start

本题的重点在于:数据段和栈段在程序加载后实际占据的空间都是以16个字节为单位的。程序中只给出了前两个字数据,其余空间都用0填充。

(1)CPU执行程序,程序返回前,data段中的数据为多少?

执行程序后,data段有16个字节空间,前两个字数据不变,其余为0。

(2)CPU执行程序,程序返回前,cs=13feh,ss=13fdh,ds=13fch.

(3)程序加载后,code段地址设为X,则data段地址为(x-2),stack段的段地址为(X-1).

(4)对于如下定义的段:

name segment

......

name ends

如果段中数据位N个字节,程序加载后,该段实际占据空间为[N/16]Byte。

 

3、编译连接下面程序,用debug加载、跟踪,回答问题。

assume cs:code,ds:data,ss:stack


code segment
start:
mov ax,stack
mov ss,ax
mov sp,16

mov ax,data
mov ds,ax

push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]

mov ax,4c00h
int 21h

code ends

data segment
dw 0123h,0456h
data ends

stack segment
dw
0,0
stack ends

end start

本题考查重点是三个段的顺序调换后是否有变化?在加载后各数据段的入口段地址必然是改变了。

(1)CPU执行程序,程序返回前,data段中的数据为多少?

运行程序至程序返回前,通过:d ds:0~f查看data段数据,观察未变。

(2)CPU执行程序,程序返回前,cs=13fch,ss=1400h,ds=13ffh.

(3)程序加载后,code段地址设为X,则data段地址为(x+3),stack段的段地址为(X+4).

4、如果不指名start入口,并且使用end替换end start,程序仍然可以执行。因为如果不指名入口,程序则从加载进内存的第一个单元起开始执行,但因为程序中有部分是作为数据使用的,如果不指明入口,CPU会把这些数值数据当成汇编指令执行,因此有必要通过start来指明入口。

5、编写code段中的代码,将a段和b段数据依次相加,结果存入c段

assume cs:code
a segment
db
1,2,3,4,5,6,7,8
a ends

b segment
db
1,2,3,4,5,6,7,8
b ends

c segment
db
0,0,0,0,0,0,0,0
c ends

code segment
start:
mov ax,a
mov ds,ax

mov ax,b
mov ss,ax

mov ax,c
mov es,ax

mov bx,0
mov cx,8
s:
mov al,ds:[bx]
add al,ss:[bx]
mov es:[bx],al
inc bx

loop s

mov ax,4c00h
int 21h

code ends
end start

6、编写code段中代码,用push指令将a段中前8个字型数据逆序存储到b段中。

assume cs:code
a segment
dw
1,2,3,4,5,6,7,8,9,0ah,0bh,0ch,0dh,0eh,0fh,0ffh
a ends

b segment
dw
0,0,0,0,0,0,0,0
b ends

code segment
start:
mov ax,a
mov ds,ax

mov ax,b
mov ss,ax
mov sp,10h

mov bx,0
mov cx,8
s:
push ds:[bx]
add bx,2

loop s

mov ax,4c00h
int 21h

code ends
end start

注:5、6题要巩固练习通过用debug命令检查内存来看程序是否执行准确。


推荐阅读
  • 线程能否先以安全方式获取对象,再进行非安全发布? ... [详细]
  • 零拷贝技术是提高I/O性能的重要手段,常用于Java NIO、Netty、Kafka等框架中。本文将详细解析零拷贝技术的原理及其应用。 ... [详细]
  • 本文回顾了作者初次接触Unicode编码时的经历,并详细探讨了ASCII、ANSI、GB2312、UNICODE以及UTF-8和UTF-16编码的区别和应用场景。通过实例分析,帮助读者更好地理解和使用这些编码。 ... [详细]
  • 单片微机原理P3:80C51外部拓展系统
      外部拓展其实是个相对来说很好玩的章节,可以真正开始用单片机写程序了,比较重要的是外部存储器拓展,81C55拓展,矩阵键盘,动态显示,DAC和ADC。0.IO接口电路概念与存 ... [详细]
  • poj 3352 Road Construction ... [详细]
  • 本文总结了一些开发中常见的问题及其解决方案,包括特性过滤器的使用、NuGet程序集版本冲突、线程存储、溢出检查、ThreadPool的最大线程数设置、Redis使用中的问题以及Task.Result和Task.GetAwaiter().GetResult()的区别。 ... [详细]
  • 单片机入门指南:基础理论与实践
    本文介绍了单片机的基础知识及其应用。单片机是一种将微处理器(类似于CPU)、存储器(类似硬盘和内存)以及多种输入输出接口集成在一块硅片上的微型计算机系统。通过详细解析其内部结构和功能,帮助初学者快速掌握单片机的基本原理和实际操作方法。 ... [详细]
  • 本文详细介绍了MySQL数据库的基础语法与核心操作,涵盖从基础概念到具体应用的多个方面。首先,文章从基础知识入手,逐步深入到创建和修改数据表的操作。接着,详细讲解了如何进行数据的插入、更新与删除。在查询部分,不仅介绍了DISTINCT和LIMIT的使用方法,还探讨了排序、过滤和通配符的应用。此外,文章还涵盖了计算字段以及多种函数的使用,包括文本处理、日期和时间处理及数值处理等。通过这些内容,读者可以全面掌握MySQL数据库的核心操作技巧。 ... [详细]
  • 在《Cocos2d-x学习笔记:基础概念解析与内存管理机制深入探讨》中,详细介绍了Cocos2d-x的基础概念,并深入分析了其内存管理机制。特别是针对Boost库引入的智能指针管理方法进行了详细的讲解,例如在处理鱼的运动过程中,可以通过编写自定义函数来动态计算角度变化,利用CallFunc回调机制实现高效的游戏逻辑控制。此外,文章还探讨了如何通过智能指针优化资源管理和避免内存泄漏,为开发者提供了实用的编程技巧和最佳实践。 ... [详细]
  • 本文提出了一种基于栈结构的高效四则运算表达式求值方法。该方法能够处理包含加、减、乘、除运算符以及十进制整数和小括号的算术表达式。通过定义和实现栈的基本操作,如入栈、出栈和判空等,算法能够准确地解析并计算输入的表达式,最终输出其计算结果。此方法不仅提高了计算效率,还增强了对复杂表达式的处理能力。 ... [详细]
  • 深入解析C语言中结构体的内存对齐机制及其优化方法
    为了提高CPU访问效率,C语言中的结构体成员在内存中遵循特定的对齐规则。本文详细解析了这些对齐机制,并探讨了如何通过合理的布局和编译器选项来优化结构体的内存使用,从而提升程序性能。 ... [详细]
  • 在C#编程中,数值结果的格式化展示是提高代码可读性和用户体验的重要手段。本文探讨了多种格式化方法和技巧,如使用格式说明符、自定义格式字符串等,以实现对数值结果的精确控制。通过实例演示,展示了如何灵活运用这些技术来满足不同的展示需求。 ... [详细]
  • Python多线程编程技巧与实战应用详解 ... [详细]
  • 多线程基础概览
    本文探讨了多线程的起源及其在现代编程中的重要性。线程的引入是为了增强进程的稳定性,确保一个进程的崩溃不会影响其他进程。而进程的存在则是为了保障操作系统的稳定运行,防止单一应用程序的错误导致整个系统的崩溃。线程作为进程的逻辑单元,多个线程共享同一CPU,需要合理调度以避免资源竞争。 ... [详细]
  • 深入解析:Synchronized 关键字在 Java 中对 int 和 Integer 对象的作用与影响
    深入探讨了 `Synchronized` 关键字在 Java 中对 `int` 和 `Integer` 对象的影响。尽管初看此题似乎简单,但其实质在于理解对象的概念。根据《Java编程思想》第二章的观点,一切皆为对象。本文详细分析了 `Synchronized` 关键字在不同数据类型上的作用机制,特别是对基本数据类型 `int` 和包装类 `Integer` 的区别处理,帮助读者深入理解 Java 中的同步机制及其在多线程环境中的应用。 ... [详细]
author-avatar
月光魔术师2702935955
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有