对数组名取地址和数组名的区别_非字符数组名给的是地址

对数组名取地址和数组名的区别_非字符数组名给的是地址以下代码会打印出什么样的日志呢? #include int a[2] = {1,2}; int main(){ printf(“a = %p
“, a); // I printf…

数组名和数组名取地址的区别

以下代码会打印出什么样的日志呢?

#include <stdio.h>
 
int a[2] = {1,2};
int main(){
        printf(“a = %p “, a); // I
        printf(“&a = %p “, &a); // II
        printf(“a + 1 = %p “, a + 1);// III
        printf(“&a + 1 = %p “, &a + 1);// IV
 
        return 0;
}
本机(linux)结果输出:
a = 0x804a014
&a = 0x804a014
a + 1 = 0x804a018
&a + 1 = 0x804a01c

没错,上面I 和 II打印出来的地址是一样的,IV 要比 III 大4个字节的地址空间。下面是我对这一现象的解释,如有不妥的地方请各位大虾一定给于指出:

首先引用《C和指针》p141中的理论:
在C中, 在几乎所有使用数组的表达式中,数组名的值是个指针常量,也就是数组第一个元素的地址。 它的类型取决于数组元素的类型: 如果它们是int类型,那么数组名的类型就是“指向int的常量指针“。
看到这里我想应该就知道为什么 会有I 和 III式的结果了。

对于II 和 IV 则是特殊情况,在《C和指针》p142中说到,在以下两中场合下,数组名并不是用指针常量来表示,就是当数组名作为sizeof操作符和单目操作符&的操作数时。 sizeof返回整个数组的长度,而不是指向数组的指针的长度。 取一个数组名的地址所产生的是一个指向数组的指针,而不是一个指向某个指针常量的指针。
所以&a后返回的指针便是指向数组的指针,跟a(一个指向a[0]的指针)在指针的类型上是有区别的。

然后我们用符号表和汇编代码来看看编译器到底是怎样区分&a 和 a, 并将其转换为汇编代码的:

通过 nm a.out 得到符号表如下:
。。。。。。。// 省略了一些与本主题无关的变量
0804a01c A _edata
0804a024 A _end
080484ec T _fini
08048508 R _fp_hw
080482bc T _init
08048330 T _start
0804a014 D a // a 变量保存在虚拟地址0x0804a014 中
0804a01c b completed.7021
0804a00c W data_start
0804a020 b dtor_idx.7023
080483c0 t frame_dummy
080483e4 T main // main函数的地址
         U printf@@GLIBC_2.0
调用gcc -S xx.c得到汇编代码:

    .file    “name_of_array.c”
.globl a
    .data
    .align 4
    .type    a, @object
    .size    a, 8 // 从这里我们便知道sizeof(a) 等于8
a:
    .long    1 // 从这里可以看出,编译器直接把 .c文件中的int 转化为long型
    .long    2
    .section    .rodata
.LC0:
    .string    “a = %p “
.LC1:
    .string    “&a = %p “
.LC2:
    .string    “a + 1 = %p “
.LC3:
    .string    “&a + 1 = %p “
    .text
.globl main
    .type    main, @function
main:
    pushl    %ebp
    movl    %esp, %ebp
    andl    $-16, %esp
    subl    $16, %esp
    movl    $.LC0, %eax // I 所对应的汇编代码
    movl    $a, 4(%esp)
    movl    %eax, (%esp)
    call    printf
    movl    $.LC1, %eax // II 所对应的汇编代码
    movl    $a, 4(%esp)
    movl    %eax, (%esp)
    call    printf
    movl    $.LC2, %eax // III 所对应的汇编代码
    movl    $a+4, 4(%esp)
    movl    %eax, (%esp)
    call    printf
    movl    $a+8, %edx // IV 所对应的汇编代码
    movl    $.LC3, %eax
    movl    %edx, 4(%esp)
    movl    %eax, (%esp)
    call    printf
    movl    $0, %eax
    leave
    ret
    .size    main, .-main
    .ident    “GCC: (Ubuntu 4.4.3-4ubuntu5) 4.4.3”
    .section    .note.GNU-stack,””,@progbits
I所对应的汇编代码 movl $a, 4(%esp)
$表示取地址,通过符号表我们知道a对应地址为0x0804a014, 所以这段代码将会打印0x0804a014。但是我们明明在代码里写的是printf(“a = %p “, a), (如果a不为数组名而是一般意义的int变量,相应的汇编码应为movl a, 4(%esp) 怎么编译后的汇编代码会是对a取地址呢? 本人猜测为编译器自动给a 加了一个取值符,从而翻译为$a。
结论: 对于用户没有明确给出&的编码,编译器翻译自动给变量a加上取值符$, 其中取a的地址得到的指针类型由数组元素决定。
II 略过

III movl $a+4, 4(%esp)
对a加上取值符得到$a,因为数组元素类型为int,所以指针每次需要移动四个字节的地址空间。 所以c代码 a + 1 翻译为汇编 $a + 4 

IV  movl $a+8, %edx 
所对应用户代码为printf(“a = %p “, &a + 1), 根据《C和指针》中的理论,当a前面有&操作符时,编译器将会把a对应符号表中的地址看作指向数组的指针,sizeof(a) 为8,
从而&a + 1 将会翻译为$a + 8
结论: 对于用户明确给出&的编码,编译器将会把取a的地址得到的指针类型看作指向数组的指针。
总结:编译器通过用户是否给出&,来决定指针变量的类型,进而翻译为相应的汇编码。 或者换句话说,&符只是用来表明变量a取地址后得到的值,被看作什么类型的指针,而不是用来表示对a进行取地址操作。

————————————————
版权声明:本文为CSDN博主「wong_daniel」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/daniel_ice/article/details/6857019

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/9810.html

(0)
上一篇 2023-01-31
下一篇 2023-01-31

相关推荐

  • redis 6.0下redis-cluster

    redis 6.0下redis-cluster伴随着Redis6.0的发布,作为最令人怦然心动的特性之一,Redis官方同时推出Redis集群的proxy了:redis-cluster-proxy,https://github.com/Redis

    2023-02-24
    162
  • Python应用领域及优势

    Python应用领域及优势Python在数据科学及机器学习领域中拥有广泛的应用。其丰富的数据科学将某些任务的执行成为可能,比如:数据采集、数据挖掘、数据分析、数据可视化及机器学习模型的构建。Python主要用于数据科学和机器学习的应用包括NumPy、SciPy、Pandas、SciKit-Learn、Keras、TensorFlow、PyTorch等库和框架。

    2024-03-11
    78
  • MySQL数据库备份和恢复「建议收藏」

    MySQL数据库备份和恢复「建议收藏」MySQL数据库备份和恢复 [toc] 备份恢复概述 为什么要备份 灾难恢复:硬件故障、软件故障、自然灾害、黑客攻击、误操作测试等数据丢失场景 备份注意要点 能容忍最多丢失多少数据 恢复数据需要在多长

    2022-12-23
    155
  • win10一键重装系统「建议收藏」

    win10一键重装系统「建议收藏」今天小编要给大家介绍的是云骑士装机大师win10一键重装系统,一键系统重装,无需电脑技术,小白在家也可自己完成安装,三步到位,安全简单!一键系统重装,0费用,0丢包,极加速!一起来看看吧。 1第一步…

    2023-04-12
    165
  • centos配置jdk_未安装设备记在建工程吗

    centos配置jdk_未安装设备记在建工程吗简介 由于我比较喜欢使用压缩包来安装,因此此处的示例都是以压缩包来示例的。同理,这并不是一篇正式的文章,所以措辞会显得有点随意且不专业。 准备 到此处选择下图中标注的 JDK 版本: 这里需要注意,因

    2023-04-16
    148
  • 成为一名高薪Python开发者的秘诀

    成为一名高薪Python开发者的秘诀要成为一名优秀的Python开发者,首先需要掌握基础的Python语法和常用的数据结构。Python作为一门解释型语言,其简洁的语法和强大的数据结构使得开发效率变得更加高效。以下是一些常用的基础语法和数据结构:

    2024-03-01
    92
  • 阿里mysql规范「终于解决」

    阿里mysql规范「终于解决」阿里mysql规范 (一)建表规约 1.【强制】表达是与否概念的字段,必须使用 is_xxx的方式命名,数据类型是 unsigned tinyint( 1表示是,0表示否),此规则同样适用于 odp…

    2023-01-31
    155
  • percona-toolkit的安装与使用

    percona-toolkit的安装与使用一、percona-toolkit的安装请参考:https://blog.csdn.net/yuanyk1222/article/details/100066788 二、下面来说说此工具的一些用法:…

    2022-12-25
    163

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注