汇编-求数组中所有偶数的和

程序重点:使用test命令来判断一个数是不是偶数,而不是使用除2求余数。

使用除2求余数的方法太占寄存器了,设置被除数需要使用eax,设置除数需要一个寄存器,假定使用ebx,两者的商会存放到eax中,同时把余数存放到edx中。所以使用这种方法使用了三个寄存器。
而使用test这种方法就使得判断变得简单。

test des,src 的指令功能就是将des和src两个操作数进行与运算。

test与and指令是不同的,因为test指令不会改变des的值,它使得CF=OF=0,并改变ZF,SF标志的值。

我们的数值在计算机中存储为二进制,所以一个数的二进制表示最低位如果是1则该数为奇数,为0则为偶数。这一点通过二进制转换十进制公式也很好证明。

因此我们将要判断的数值与1进行test操作,如果是偶数则结果是0,通过判断ZF数值就可以判断是不是偶数。

下面是源程序:

;功能:求数组中所有偶数之和
;作者:王文堃
;创建时间:2016/4/19

INCLUDE vcIO.inc
.data
    array DWORD 12,34,123,78,43,234,79,86,98,20
    esum DWORD ? ;记录总和
    str_output BYTE "数组中所有偶数的和为:%d",0ah,0
.code
main PROC
    xor ebx, ebx ;使用寄存器存放总和,最后再赋值给esum以减少访存次数
    mov edi, LENGTHOF array-1 ;edi记录数组最后一个数

    jmp TESTING
    ;判断是不是偶数,若是则加入总和
DOLOOP:
    test array[edi*4], 1
    jnz NEXT
    add ebx, array[edi*4]
NEXT:
    dec edi

TESTING:
    cmp edi, 0
    jge DOLOOP 
    ;编程要点,这里要使用有符号的大于等于跳转。
    ;如果使用无符号跳转jae则在递减的时候减到0之后还会减成全为F的最大数,陷入死循环。

    ;输出
    mov esum, ebx
    invoke printf, offset str_output, esum
    ret
main ENDP
    ;子程序
END main

需要说明的是,为了减少寄存器的使用,我们使用了ecx倒着访问数组,判断条件是ecx大于等于0就循环。

但是一定要使用有符号数跳转jge,如果使用无符号跳转就会死循环。
下面解释一下为什么会死循环,其实原因也十分简单。

那无符号的八位数来说,它能表示的范围是0~255,如果将0-1结果就是255。所以当ecx减到0的时候,下一次再减1的时候它就为8个F的这样一个最大的数,它永远不可能因为ecx小于0而停止循环。

  • 0
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
5.1试编写一个汇编语言程序,要对键盘输入的小写字母用大写字母显示出来。 5.2 编写程序,从键盘接收一个小写字母,然后找出它的前导字符和后续字符,再按顺序输出 5.3 将AX寄存器的16位数分成4组,每组4位,然后把这四组数分别放在AL、BL、CL、DL。 5.4 试编写一程序,要比较两个字符串STRING1和STRING2所含字符是否相同,若相同则显示‘MATCH’,若不相同则显示‘NOT MATCH’。 5.5 试编写一程序,要能从键盘接收一个个位数N,然后响铃N次。 5.6 编写程序,将一个包含有20个数据的数组M分成两个数组:正数数组P和负数数组N,并分别把这两个数组的数据的个数显示出来。 5.7 试编制一个汇编语言程序,出首地址为DATA的100D字数组的最小偶数,并把它放在AX。 5.8 把AX存放的16位二进制数K看作是8个二进制的“四分之一字节”。试编写一个程序,要数一下值为3(即11B)的四分之一字节数,并将该数在终端上显示出来。 5.9 试编写一汇编语言程序,要从键盘接收一个四位的十六进制数,并在终端上显示与它等值的二进制数。 5.10 设有一段英文,其字符变量名为ENG,并以$字符结束。试编写一程序,查对单词SUN在该文的出现次数,并以格式”SUNXXXX“显示出次数。 5.11 从键盘输入一系列以$为结束的字符串,然后对其的非数字字符进行计数,并显示出计数结果。 5.12 有一个首地址为MEM的100D字数组,试编制程序删除数组所有为零的项,并将后续项向前压缩,最后将数组的剩余部分补上零。 5.13 在STRING到STRING+99单元存放着一个字符串,试编制一程序测试该字符串是否存在数字,如有,则把CL的第五位置1,否则置0. 5.14 在首地址为TABLE的数组按递增次序存放着100H个16位补码数,试编写一个程序把出现次数最多的数及其出现的次数分别放在AX和CX。 5.15 数据段已定义了一个有N个字数据的数组M,试编写一程序出M绝对值最大的数,把它放在数据段的M+2n单元,并将该数的偏移地址存放在M+2(n+1)单元。 5.16 在首地址为DATA的字数组,存放了100H个16位补码数,试编写一个程序出它们的平均值放在AX寄存器;并数组有多少个数小于此平均值,将结果放在BX寄存器。 5.17 试编写一个程序,把AX的十六进制数转换为ASCII码,并将对应的ASCII码依次存放到MEM数组的四个字节,例如:当(AX)=2A49H时,程序执行完后,MEM的4个字节的内容为39H,34H,41H和32H。 5.18 把0~100D之间的30个数存入以GRADE为首地址的30个字数组,GRADE+i表示学号i+1的学生的成绩。另一个数组RANK为30个学生的名次表,其RANK+i的内容是学号为i+1的学生的名次。编写一程序,根据GRADE的学生成绩,将学生名次填入RANK数组。 5.19 已知数组A包含15个互不相等的整数,试编写一程序,把既在A又在B出现的整数存在于数组C。 5.20 设在A,B和C单元存放着三个数,若三个数都不是0,则出三树之和并存放于D单元;其有一个数为0,则把其他两个数也清零。试编写此程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值