大家好,我是汤姆凯特。
文章目录
☀️作者简介:大家好我是汤姆凯特,大家可以叫我汤姆
🐋个人主页:IM汤姆凯特的CSDN博客
🎁系列专栏:【ARM嵌入式基础】
🌱每日一句:
“人生的道路都是由心来描绘的。所以,无论自己处于多么严酷的境遇之中,心头都不应为悲观的思想所萦绕。”——稻盛和夫
【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
但是结果却出错了👇
这是为什么呢?我们来看一下之前讲scanf用法时的例子👇
这里可以看到,通过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往内存中传值,必须要给出明确的地址,而不是地址中存放的数据
- 再使用跳转指令后,一定要给使用过的寄存器重新赋值