汇编 --- 栈结构的妙用

在js中双重循环,代码如下:

for(let i =0; i < l1;i++){for(let j=0; j< l2;j++){// todo...}
}

以上代码在汇编中是如何实现的呢.

  • 汇编中用cx和loop来代表一次循环
	mov cx, 4
s:	mov ax,1loop s
  • 但是如果使用如下方法实现双重循环将会出错
assume cs:codesg, ds:datasgdatasg segment
db	'ibm             '
db	'dec			 '
db	'dos			 '
db	'vax			 '
datasg endscodesg	segment
start:	mov	ax, datasgmov ds,	axmov bx,	0		; 用BX来定位行mov	cx,	4s0:	mov	si,	0		; 用si来定位列mov	cx,	3s:	mov	al,	[bx+si]and	al,	11011111bmov	[bx+si], alinc siloop s		    ; 此时的 cx 已经为0add	bx, 16loop s0			; cx = cx -1 , 再判断 cx 为是否为0mov ax, 4c00hint 21hend codesg
end start
  • 原因如下:
    • 每当执行 loop 语句时, 实际是执行 cx = cx -1 然后判断 cx 是否为0. 于是在内层循环后(loop s), cx =0 , 然后再到 loop s0时,此时先执行 cx = cx - 1, 即此时 cx = FFFF 因此会陷入死循环
  • 改进办法.用寄存器dx来保存进入内层循环的cx,然后再内存循环结束时,将寄存器dx中的值赋给cx
start:	mov ax,	datasgmov ds,	axmov bx,	0mov cx, 4s0:	mov dx, cxmov	si, 0mov cx, 3s:		mov al, [bx+si]and	al,	11011111bmov	[bx+si],	alinc siloop sadd bx, 16mov cx, dxloop s0mov ax, 4c00hint 21h
  • 以上方法可以解决两层循环的问题,但是CPU中的寄存器毕竟是有限的,当循环次数多的时候,寄存器将不够用.
  • 考虑到内存,可以将寄存器cx的值存入内存中.然后在内存循环结束后,在从内存中读取值给cx
assme cs:codesg, ds: datasgdatasg segment
; 其他代码略
dw 	0			; 定义一个字单元, 用来保存cx
datasg endscodesg segment
start:		mov ax, datasgmov ds, ax				; 汇编中用ds来定位数据地址mov bx, 0				; 偏移量为0mov cx, 4				; 外层循环为4s0:		mov ds:[40H], cx		; 将外层循环的次数保存到内存datasg:40H单元中mov si, 0mov cx, 3s:		mov al, [bx+si]and al, 11011111bmov [bx+si], alinc siloop sadd bx, 16				; 移到下一个字单元mov cx, ds:[40H]		; 从内存中取出当前循环的次数loop s0mov ax,	4c00hint 21h
end codesg
end start
  • 以上方法可以解决CPU中寄存器不够用的情况,但对于保存的多个数据,程序猿们必须要记住数据保存在哪个内存单元中.当代码量大的时候,需要写很多注释,也不利于阅读与维护
  • 看VC++ 6.0编译器是如何处理的
    在这里插入图片描述
  • 通过反编译,查看汇编源码可以发现.
  • 在遇到函数时,编译器将当前环境push进一个栈中,当执行完毕,将栈中的环境pop出来
  • 于是上面的代码可以改为如下:
stack segment			; 栈空间dw 0,0,0,0,0,0,0,0	; 16字节(根据需要定)
stack endscodesg segment
start:	mov ax, stacksgmov ss, axmov sp,	16		; 汇编中ss 指向栈端, sp代表偏移量为16mov ax, datasgmov ds, axmov bx, 0mov cx, 4s0:	push cxmov si,0mov cx, 3s:	mov al, [bx+si]and al, 11011111bmov [bx+si], alinc siloop sadd bx, 16pop cxloop s0mov ax, 4c00hint 21h
codesg ends
end start

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

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

相关文章

PHP 安全编程建议

PHP 安全编程建议 简介 要提供互联网服务&#xff0c;当你在开发代码的时候必须时刻保持安全意识。可能大部分 PHP 脚本都对安全问题都不在意&#xff0c;这很大程度上是因为有大量的无经验程序员在使用这门语言。但是&#xff0c;没有理由让你因为对你的代码的不确定性而导致不…

day2-列表、元组、字典、字符串

1.列表&#xff08;list&#xff09; names[悟空,艾玛,克林,龟仙人,天津饭,饺子,乌龟] print(names)---》列表&#xff0c;然后打印。 names[悟空,艾玛,克林,龟仙人,天津饭,饺子,乌龟]# list section section1names[0] #下标从0开始 #print(section1) #结果为悟空seciton2names…

开漏输出、推挽输出的区别

推挽输出:可以输出高,低电平,连接数字器件。 输出 0 时&#xff0c;N-MOS 导通&#xff0c;P-MOS 高阻&#xff0c;输出0。 输出 1 时&#xff0c;N-MOS 高阻&#xff0c;P-MOS 导通&#xff0c;输出1&#xff08;不需要外部上拉电路&#xff09;。 开漏输出:输出端相当于三极管…

罗马数字

古罗马帝国开创了辉煌的人类文明&#xff0c;但他们的数字表示法的确有些繁琐&#xff0c;尤其在表示大数的时候&#xff0c;现在看起来简直不能忍受&#xff0c;所以在现代很少使用了。之所以这样&#xff0c;不是因为发明表示法的人的智力的问题&#xff0c;而是因为一个宗教…

汇编 --- 从磁盘(扇区2到18)上读取数据到内存中

下面代码读取柱面:0,磁头:0,扇区从2到18的数据到内存 0x8200~0xa3ff处 需要明白以下几点: 给定柱面,磁头,一个扇形区域是512字节,对应的物理可以理解为512个灯泡组(一个灯泡组有8个小灯泡)确定读取到内存中的位置 为什么是0x8200:因为0x8000~0x81ff这512个字节要留给启动区.为…

那些值得思考的PHP问题

那些值得思考的PHP问题 1、关于弱类型 函数strpos是返回字符串str2在str1的位置&#xff0c;没有找到则返回false&#xff0c;然而如果在实际应用上返回的位置是0&#xff0c;在if语句中0也被当作false&#xff0c;所以我们需要对false做类型判断&#xff0c; $str1 yabadaba…

Lvs Tun隧道模式配置

######## TUN是IP Tunneling &#xff0c;IP隧道的简称&#xff0c;它将调度器收到的IP数据包封装在一个新的IP数据包中&#xff0c;转交给应用服务器&#xff0c;然后实际服务器的返回数据会直接返回给用户。 工作原理&#xff1a; 用户请求负载均衡服务器&#xff0c;当IP数…

mysql-常用sql

记录下工作中常用的sql 删除重复数据 delete from student where id not in (select min(id) from student group by name); -- 该语句在mysql下会报错&#xff0c; -- 执行报错&#xff1a;1093 - You cant specify target table student for update in FROM clause -- 原因是…

优雅的使用Laravel之phpstorm配置

优雅的使用Laravel之phpstorm配置 先打开一个Laravel 项目&#xff0c;然后在project tool 窗口选择根节点、然后右键->Composer | Init composer 。 如果你的电脑里没有composer.phar&#xff0c;可以点击链接来下载。然后点击项目 composer->add denpendency.. 搜索ba…

算法 --- 求两个集合的并集

const unionL (l1, l2)>{for(let i0; i <l1.length; i){if(l2.indexOf(l1[i]) -1){l2.push(l1[i])}}return l2 }let l1 [1,2,3,4]; let l2 [1,2,5]; console.log(unionL(l1,l2)); //[1, 2, 5, 3, 4]

java实验四——找鞍点

package hello;public class 实验四 {public static void main(String[] args) {// TODO Auto-generated method stubint[][] a {{9,8,6},{2,3,1},{8,5,2}};System.out.println("二维数组为&#xff1a;");for(int i0;i<a.length;i){for(int j0;j<a[i].length;…

BZOJ 4551树题解

好吧&#xff0c;洛谷的数据比较水暴力就可以过。。。。&#xff08;而且跑到飞快&#xff09; 不过&#xff08;BZ水不过去&#xff09;还是讲讲正规的做法。 其实一眼可以看出可以树剖&#xff0c;但是&#xff0c;码起来有点麻烦。 其实有一种更简单的离线做法。 我们很容易…

es6 --- 使用Symbol保护私有变量

定义一个人物类 假设其属性有姓名和性别我们希望,性别在声明后就固定不变 传统方法 var Person (function(){var _gender ;function P(name, gender){this.name name;_gender gender;}P.prototype.getGender function(){return _gender;}return P; })();var p1 new Pe…

组合数

long long factorial(int n) {long long m 1;for(int i1;i<n;i)m*i;return m; } long long C(int n,int m) {return factorial(n)/(factorial(m)*factorial(n-m));//可能会溢出 } 正解&#xff1a; long long C(int n,int m) {if(m<n-m) m n-m;long long ans 1;for(in…

Mysql中的联合索引、前缀索引、覆盖索引

Mysql中的联合索引、前缀索引、覆盖索引 索引 索引是一种特殊的文件&#xff0c;它们包含着对数据表里所有记录的引用指针。更通俗的说&#xff0c;数据库索引好比是一本书前面的目录&#xff0c;能加快数据库的查询速度。 联合索引 又名复合索引&#xff0c;由两个或多个列…

LVM逻辑卷管理

什么是逻辑卷&#xff1f;因为可以将文件系统像卷一样伸长或缩短之故。 LVM做法&#xff0c;将几个物理分区或磁盘&#xff0c;通过软件组合成为一块看起来是独立的大磁盘&#xff08;VG&#xff09;&#xff0c;然后将这块大磁盘再经过分成可使用分区&#xff08;LV&#xff0…

es6 --- 自制迭代器

对象 对象如下 const obj {left: 100,top: 200 }不可迭代 for(let attr of obj){console.log(attr); }迭代规则 可迭代,所具有的属性[Symbol.iterator] 需要自己给obj添加迭代规则 obj[Symbol.iterator] () >{// 获取obj的所有键let keys Object.keys(obj);let len …

软件工程的实践项目课程的自我目标

对实践项目完成后学习到的能力的预期&#xff1a;这算是自己第一次参加的团队合作的软件实践吧&#xff0c;以前自己搞的小“玩意”&#xff0c;实难登大雅之堂&#xff0c;期待实践项目后--->1、自己的代码能力能够有较明显的提高&#xff0c;代码更加规范。 2、提升团队合…

[Shell] swoole_timer_tick 与 crontab 实现定时任务和监控

手动完成 "任务" 和 "监控" 主要有下面三步&#xff1a; 1. mission_cron.php&#xff08;定时自动任务脚本&#xff09;&#xff1a; <?php /*** 自动任务 定时器 (5s 执行).** swoole_timer_tick 解决秒级定时&#xff1b;* 如需调整&#xff0c;注意…

关于项目调研

一、柠檬时代app 1、作品内容&#xff1a; 该作品主要为每一所高校的大学生量身定制的手机生活助手&#xff0c;由您自主运营的校园手机客户端。开放的自定义平台&#xff0c;汇聚本校周边所有生活服务、外卖商家、娱乐休闲、新鲜事儿、知名社团、周边商铺、校园达人等栏目。 …