因为数组的存储也只能存放到内存中,所以要在全局变量中定义一个给定长度的数组 //定义一个长度为5的数组
ary:.word 0,0,0,0,0
第三步、构建第一个循环
第一个循环是要实现键入的数据存到数组中,那么需要用到scanf,首先把键入的值存到数组的首地址,然后向后寻址下一个元素的。
错误示范 用我们很熟悉的遍历数组的循环看一下 将数组的首地址读取到r5中,在循环中将r5的值传给r1,并向后取址 ldr r5, =ary
b testfor
loop:
ldr r0,=fmt1
ldr r1,[r5],
bl scanf
add r4,
testfor:cmp r4,
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,
bl scanf
add r4,
testfor:cmp r4,
blt loop
第四步、构建第二个循环 第二个循环就是之前常用的遍历数组,看不明白下面代码的可以看这一篇《ARM循环遍历数组》 b testfor2
loop2:
ldr r0,=fmt
ldr r1,[r5],
bl printf
add r4,
testfor2:cmp r4,
blt loop2
这里需要特别注意的是
1.这里再次调用r4,但是上面循环,已经将循环的变量r4递增成了5,所以这里需要再次将r4归零
2.这里运用了多次B跳转指令,所以要给用到的r5,重新取一次数组的首地址。
mov r4,
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,
ldr r5, =ary
b testfor
loop:
ldr r0,=fmt1
//ldr r1,[r5],
mov r1,r5
add r5,
bl scanf
add r4,
testfor:cmp r4,
blt loop
mov r4,
ldr r5, =ary // 重新赋值
b testfor2
loop2:
ldr r0,=fmt
ldr r1,[r5],
bl printf
add r4,
testfor2:cmp r4,
blt loop2
ldr r0,=fmt0
bl printf
mov r0,
ldmfd sp!,{lr}
mov pc, lr
.end
尝试运行:
输入"64、-8、0、91、-210"这五个数进行尝试
总结:
- 将一串键入的数存到数组中,需要单独用一个循环
- 想要用scanf往内存中传值,必须要给出明确的地址,而不是地址中存放的数据
- 再使用跳转指令后,一定要给使用过的寄存器重新赋值
本期就结束了,如果对您有帮助,点赞+评论支持一下博主再走吧
还没有关注汤姆的朋友,点个关注每天学一点汇编
下期预告:【ARM汇编数组】如何对键入的数据做判断(下)
|
|
评论(0)