大家好,我是汤姆凯特。
☀️作者简介:大家好我是汤姆凯特,大家可以叫我汤姆
个人主页:IM汤姆凯特的CSDN博客
系列专栏:【ARM嵌入式基础】
每日一句:
“人生的道路都是由心来描绘的。所以,无论自己处于多么严酷的境遇之中,心头都不应为悲观的思想所萦绕。”——稻盛和夫
前面一篇文章说到了,实时判断虽然可以判断,但是那穿数并没有存放起来,比较完也没有意义,并且输入完5个循环后最后还得键入一个数。因此这里带大家学习如何将键入的数据存到数组中。
写汇编之前可以先用C的伪代码表示出来,便于理清要实现的功能和逻辑 这里我们可以看到,这里有两个循环,第一个循环为了将键入的数据存储到数组中,第二个循环把数组中的数据遍历出来。所以我们用汇编编写可以参考同样的方式。 |
#include
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"这五个数进行尝试