汇编实现可见ASCII码值转字符并输出

文章目录

  • 实现思路
  • 代码实现
    • 方法1:一次性输出所有字符
    • 方法2:循环输出
  • 总结

实现思路

汇编实现可见ASCII码值转字符并输出_第1张图片

每一个字符都对应一个ASCII码值,调用dispc函数就可以将数字输出成对应的字符。那么,已知的可见字符取值范围是20H~7FH(最后一个字符透明),这里就用两种可以实现的思路了:

1.定义一个字节数组,存放20H~7FH所有的数,调用dispmsg函数直接输出整个数组

2.通过循环体,每次只输出一个字符

代码实现

代码是在MASM编译器下实现的,具体能不能在其他编译器上实现就不太清楚了,但是具体的语句的功能都附加上了注释,大体上的思路是一致的。

方法1:一次性输出所有字符

include io32.inc ;包含32为输入输出文件
    .data ;数据段
    ;ASCII值数组的手动填充,其中除了目标字符之外,还有空格、分隔符、表头、换行符等附加信息,都使用ASCII码值表示了(有点臃肿,没办法)
    table byte 20H,20H,20H,20H,7CH,20H,30H,20H,31H,20H,32H,20H,33H,20H,34H,20H,35H,20H,36H,20H,37H,20H,38H,20H,39H,20H,41H,20H,42H,20H,43H,20H,44H,20H,45H,20H,46H,0DH,0AH
          byte 2DH,2DH,2DH,20H,2BH,20H,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,2DH,0DH,0AH
          byte 20H,33H,30H,20H,7CH,20H,20H,20H,21H,20H,22H,20H,23H,20H,24H,20H,25H,20H,26H,20H,27H,20H,28H,20H,29H,20H,2AH,20H,2BH,20H,2CH,20H,2DH,20H,2EH,20H,2FH,0DH,0AH
          byte 20H,34H,30H,20H,7CH,20H,30H,20H,31H,20H,32H,20H,33H,20H,34H,20H,35H,20H,36H,20H,37H,20H,38H,20H,39H,20H,3AH,20H,3BH,20H,3CH,20H,3DH,20H,3EH,20H,3FH,0DH,0AH
          byte 20H,35H,30H,20H,7CH,20H,40H,20H,41H,20H,42H,20H,43H,20H,44H,20H,45H,20H,46H,20H,47H,20H,48H,20H,49H,20H,4AH,20H,4BH,20H,4CH,20H,4DH,20H,4EH,20H,4FH,0DH,0AH
          byte 20H,36H,30H,20H,7CH,20H,50H,20H,51H,20H,52H,20H,53H,20H,54H,20H,55H,20H,56H,20H,57H,20H,58H,20H,59H,20H,5AH,20H,5BH,20H,5CH,20H,5DH,20H,5EH,20H,5FH,0DH,0AH
          byte 20H,37H,30H,20H,7CH,20H,60H,20H,61H,20H,62H,20H,63H,20H,64H,20H,65H,20H,66H,20H,67H,20H,68H,20H,69H,20H,6AH,20H,6BH,20H,6CH,20H,6DH,20H,6EH,20H,6FH,0DH,0AH
          byte 20H,38H,30H,20H,7CH,20H,70H,20H,71H,20H,72H,20H,73H,20H,74H,20H,75H,20H,76H,20H,77H,20H,78H,20H,79H,20H,7AH,20H,7BH,20H,7CH,20H,7DH,20H,7EH,20H,7FH,0DH,0AH
    .code ;代码段
;程序起始位置
start:
    xor eax,eax ;清零eax寄存器
    mov eax,offset table ;将ASCII码值数组的首地址放入eax寄存器中
    call dispmsg ;输出字符
exit 0 ;程序正常执行终止
end start ;汇编结束

运行结果

汇编实现可见ASCII码值转字符并输出_第2张图片

上面的执行指令也可以是out\test2.exe

符合预期输出

方法2:循环输出

由于循环输出不太好控制输出表头,这里就不再考虑了(需要的同学可以自己加上,很麻烦滴~)。

下面的注释很清楚了,就不再摘出来解释了,注释结合代码很好理解的。重点在于循环指令loop以及配合使用充当计数器的ecx寄存器,这些初学者可能会有一点点吃力,不过可以去网上或者课本或者课本上翻一下就能找到相关知识点。(作为初学者的我吃了一个小小的苦头)

include io32.inc ;包含32为输入输出文件
.data ;数据段
    
    
.code ;代码段
;程序起始位置
start:
    xor ecx,ecx ;清零寄存器ecx
    xor eax,eax ;清零寄存器eax
    mov ecx,00000060H ;ecx寄存器充当计数器,为0之后停止循环
    mov eax,00000020H ;eax寄存器放的是字符的ASCII码值,初始值为20H(空格的值)
    mov ebx,0 ;ebx寄存器存放 换行计数
    again:
        ;输出字符和空格
        call dispc ;dispc将eax寄存器中的ASCII码值以字符的形式输出来,相当于c语言中的printf("%c",eax);
        mov edx,eax ;为了使用eax寄存器输出空格,先将eax的ASCII码值存到edx寄存器中
        mov eax,20H ;将空格的ASCII码值赋值给eax寄存器
        call dispc ;输出空格
        mov eax,edx ;再将保存在edx寄存器中的ASCII码值放入eax中
        
        ;计数器和字符ASCII码值更新
        add eax,1 ;字符ASCII码值加1,在下一次循环中输出下一个字符
        add ebx,1 ;换行计数加1
        cmp ebx,16 ;若ebx-16为0(满足一行16个字符),设置zf=0
        jz crlf ;zf=0时,进入分支语句:crlf(分支名)
    loop again ;走到这里,ecx自减1(规定!!!,指定的就是ecx寄存器),进入下次循环,从again处继续执行代码

    ;分支语句的定义
    crlf:  
        call dispcrlf ;换行
        xor ebx,ebx ;将换行计数器清零
        loop again ;回到循环体起始位置,进入下次循环,从again处继续执行代码
exit 0 ;程序正常执行终止
end start ;汇编结束

运行结果

汇编实现可见ASCII码值转字符并输出_第3张图片

也是符合预期的。

下面是我的文件管理,其中源文件和生成文件的默认位置都应该是在D:\MASM下,我改过之后源文件就放在了progs中,生成文件就在out中。这个大家按照默认的来就行,有需要的管理起来的可以在评论区留言,我再给出解决方案。

汇编实现可见ASCII码值转字符并输出_第4张图片

总结

总的来说,这个就是对ASCII码值与字符之间的关系的知识点考察,实际并不难处理,通过编码也加深了对知识点的掌握能力。

你可能感兴趣的:(汇编,汇编)