【嵌入式ARM汇编】如何将键入的数据存到数组中?

IM汤姆凯特 发表于 2022/04/24 09:13:50 2022/04/24
【摘要】 嵌入式ARM汇编中,如何将一串数据传到数组中?汇编调用C函数的练习、汇编中循环语句的练习。

每日一句:

“人生的道路都是由心来描绘的。所以,无论自己处于多么严酷的境遇之中,心头都不应为悲观的思想所萦绕。”——稻盛和夫


【ARM汇编数组】如何将键入的数据存到数组中

写在前面:

前面一篇文章说到了,实时判断虽然可以判断,但是那穿数并没有存放起来,比较完也没有意义,并且输入完5个循环后最后还得键入一个数。因此这里带大家学习如何将键入的数据存到数组中。

第一步、用C的伪代码表示

写汇编之前可以先用C的伪代码表示出来,便于理清要实现的功能和逻辑

这里我们可以看到,这里有两个循环,第一个循环为了将键入的数据存储到数组中,第二个循环把数组中的数据遍历出来。所以我们用汇编编写可以参考同样的方式。
#include<stdio.h>
int main(){
    int ary[10];
    int i;
    int a;
    for ( i = 0; i < 10; i++)
    {
       scanf("%d",&a);
       ary[i] = a;
    }
    for (i = 0;i < 10;i++){
    	printf("%3d",ary[i]);
	}
    return 0; 
}

第二步、给数组分配空间

因为数组的存储也只能存放到内存中,所以要在全局变量中定义一个给定长度的数组
//定义一个长度为5的数组
    ary:.word 0,0,0,0,0

第三步、构建第一个循环

第一个循环是要实现键入的数据存到数组中,那么需要用到scanf,首先把键入的值存到数组的首地址,然后向后寻址下一个元素的。

错误示范

用我们很熟悉的遍历数组的循环看一下

将数组的首地址读取到r5中,在循环中将r5的值传给r1,并向后取址

    ldr r5, =ary        
    b testfor
loop:   
        ldr r0,=fmt1
        ldr r1,[r5],#4 
        bl scanf

        add r4,#1
testfor:cmp r4,#5
        blt loop

但是结果却出错了:point_down:

在这里插入图片描述

这是为什么呢?我们来看一下之前讲scanf用法时的例子:point_down:

在这里插入图片描述

这里可以看到,通过scanf键入到数据后,其实是只需要把指定位置的地址给传参寄存器就可以,根本不需要取地址里面的内容。

所以这里用 ldr r1,[r5],#4 肯定是错的,正确的应该是下面这段代码

正确代码

r5首先是指向内存中定义数组的首地址,将r5的地址给r1,然后通过scanf把值放到r5存的内存地址中

现在r5里面存的是一个地址,然后给r5自增4,就指向了数组下一个元素对应的地址,这样才实现循环往数组中存数的功能
    ldr r5, =ary        
    b testfor
loop:   
        ldr r0,=fmt1
        mov r1,r5
        add r5,#4
        bl scanf
        add r4,#1
testfor:cmp r4,#5
        blt loop

第四步、构建第二个循环

第二个循环就是之前常用的遍历数组,看不明白下面代码的可以看这一篇《ARM循环遍历数组》

     b testfor2
  loop2:
         ldr r0,=fmt
         ldr r1,[r5],#4
         bl printf  
         add r4,#1
 testfor2:cmp r4,#5
         blt loop2

这里需要特别注意的是

1.这里再次调用r4,但是上面循环,已经将循环的变量r4递增成了5,所以这里需要再次将r4归零


2.这里运用了多次B跳转指令,所以要给用到的r5,重新取一次数组的首地址。

     mov r4,#0
     ldr r5, =ary // 重新赋值

第五步、完善指令

核心代码完成之后,看一下我们用到了哪些寄存器的值,给寄存器和格式串初始化。

完整源代码:

.data
    fmt1:.asciz "%d"
    fmt:.asciz "%10d\n"
    ary:.word 0,0,0,0,0

.text
.globl main
 main:
    stmfd sp!,{lr}
    mov r4,#0
    ldr r5, =ary        
    b testfor
loop:   
        ldr r0,=fmt1
        //ldr r1,[r5],#4  取的的不是地址!
        mov r1,r5
        add r5,#4
        bl scanf

        add r4,#1
testfor:cmp r4,#5
        blt loop
    
     mov r4,#0
     ldr r5, =ary // 重新赋值
     b testfor2
  loop2:
         ldr r0,=fmt
         ldr r1,[r5],#4
         bl printf  
         add r4,#1
 testfor2:cmp r4,#5
         blt loop2
	    ldr r0,=fmt0
        bl printf
    mov r0, #0
    ldmfd sp!,{lr}
    mov pc, lr
.end

尝试运行:

输入"64、-8、0、91、-210"这五个数进行尝试

在这里插入图片描述


总结:

  • 将一串键入的数存到数组中,需要单独用一个循环
  • 想要用scanf往内存中传值,必须要给出明确的地址,而不是地址中存放的数据
  • 再使用跳转指令后,一定要给使用过的寄存器重新赋值

本期就结束了,如果对您有帮助,点赞+评论支持一下博主再走吧
还没有关注汤姆的朋友,点个关注每天学一点汇编
下期预告:【ARM汇编数组】如何对键入的数据做判断(下)

在这里插入图片描述

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区),文章链接,文章作者等基本信息,否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件至:cloudbbs@huaweicloud.com进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容。
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。