【C语言学习】数组名的实质

2023-10-16 23:20
文章标签 语言 学习 实质 数组名

本文主要是介绍【C语言学习】数组名的实质,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

🐱作者:一只大喵咪1201
🐱专栏:《C语言学习》
🔥格言:你只管努力,剩下的交给时间!
图

数组名的本质

  • 描述
  • 一维数组
    • 在内存中的储存
    • 一般情况下数组名的实质
    • 在操作符sizeof下数组名的实质
    • 在操作符&下数组名的实质
  • 二维数组
    • 在内存中的储存
    • 一般情况下数组名的实质
    • 在操作符sizeof下数组名的实质
    • 二维数组行数和列数的求法:
    • 在操作符&下数组名的实质
  • 总结

描述

在需要创建很多相同类型的元素时就要使用到数组,而创建数组就需要给这个数组起个名字,但是这个名字有一定的意义,下面本喵来详细说一下。

一维数组

在内存中的储存

先创建一个一维数组。
代码如下:

#include<stdio.h>int main()
{int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };return 0;
}

数组在内存中是这样存储的:
数组存放
每个元素所在的地址如下:

#include<stdio.h>int main()
{int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };int i = 0;for (i = 0; i < 10; i++){printf("&arr[%d] = %p\n", i, &arr[i]);}return 0;
}

运行结果:
代码运行结果
可以看到,数组中的每个元素都有一个地址,而且是从低到高,就像是这些元素住在一栋楼里,每个元素占一层,由于创建的是整型数组,每层的大小是4个字节,而数组名就相当于这栋楼,在数组名后加上下标就可以找到对应的元素,就像是门牌号。

一般情况下数组名的实质

#include<stdio.h>int main()
{int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };printf("%p\n", &arr[0]);//打印首元素的地址printf("%p\n", arr);//打印数组名的地址return 0;
}

运行结果
代码运行结果
可以看到,数组名的本质就是数组首元素的地址。

在操作符sizeof下数组名的实质

sizeof操作符的作用是求出数据所占内存空间的大小。当数组名遇到了sizeof时就不是代表数组首元素地址了,而是代表的整个数组,就像上诉所说的整栋楼。

#include<stdio.h>int main()
{int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };int sz = sizeof(arr);//求数组大小printf("sizeof(arr) = %d\n", sz);return 0;
}

运行结果:
代码运行结果
可以看到结果是40,由于创建的是整型数组,里面又10个元素,每个元素的大小是4个字节,所以得出的结果是40。
此时数组名代表的就是整个数组。

在操作符&下数组名的实质

&是取地址操作符,就是得到所创建变量在内存中的地址。
当数组名遇到&时的实质会是什么呢?

#include<stdio.h>int main()
{int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };printf("%p\n", &arr[0]);//打印首元素地址printf("%p\n", &arr[0] + 1);//打印首元素地址加1printf("\n");printf("%p\n", &arr);//打印数组名取地址printf("%p\n", &arr + 1);//打印数组名取地址加1return 0;
}

运行结果:
代码运行结果
首元素地址加1的结果就是在首元素地址的基础上加了4个字节。
数组名取地址加1的结果是在首元素地址的基础上加了40个字节。
虽然首元素地址是数组名地址的表现相同,但是实质不同。
首元素取地址的打印结果是首元素地址,但是数组名取地址代表的是整个数组的地址,当加1后直接加了40个字节,也就是加了整数组元素所占的内存。
此时数组名代表的就是整个数组。

二维数组

在内存中的储存

先创建一个二维数组。
代码如下:

#include<stdio.h>int main()
{int arr[3][4] = { {1,2,3},{3,4,5,6},{1,2} };//创建二维数组,三行四列,空白部分自动用0填充return 0;
}

二维数组在内存中是这样储存的:
储存
每个元素所在的地址如下:

#include<stdio.h>int main()
{int arr[3][4] = { {1,2,3},{3,4,5,6},{1,2} };//创建二维数组,三行四列,空白部分自动用0填充int i = 0;int j = 0;for (i = 0; i < 3; i++)//外循环控制行{for (j = 0; j < 4; j++)//内循环控制列printf("&arr[%d][%d] = %p\n", i, j, &arr[i][j]);printf("\n");}return 0;
}

运行结果:
代码运行结果
可以看到,数组中的每个元素都有一个地址,而且是从低到高,同样可以理解为这些元素住在一栋楼里,每个元素占一成,大小是四个字节,在数组名后加上下标就可以找到具体的某个元素。
注意
二维数组可以看出是一维数组,一维数组的成员是arr[0],arr[1],arr[2],即数组的每一行看成是一个成员,就像在楼里的每4个元素看成一户,一共有3户,就是12个元素。

一般情况下数组名的实质

#include<stdio.h>int main()
{int arr[3][4] = { {1,2,3},{3,4,5,6},{1,2} };//创建二维数组,三行四列,空白部分自动用0填充printf("%p\n", &arr[0][0]);//打印首元素地址printf("%p\n", &arr[0][0] + 1);//打印首元素地址加1printf("\n");printf("%p\n", arr);//打印数组名取地址printf("%p\n", arr + 1);//打印数组名取地址加1return 0;
}

运行结果:
代码运行结果
数组元素首地址加1后地址多了4个字节。
数组名加1后地址多了16个字节,因为一行元素是4个,所占空间大小是16个字节。
可以看到,二维数组名的本质是首行元素的地址,但是表现出来的值与首元素地址相同。

在操作符sizeof下数组名的实质

sizeof操作符的作用上面已经介绍过。当二维数组名遇上sizeof也是代表的整个数组,二维数组的行名arr[i]遇上sizeof代表的这一行。

#include<stdio.h>int main()
{int arr[3][4] = { {1,2,3},{3,4,5,6},{1,2} };//创建二维数组,三行四列,空白部分自动用0填充int sz1 = sizeof(arr);//求整个数组的大小int sz2 = sizeof(arr[0]);//求第一行的大小printf("sizeof(arr) = %d\n", sz1);printf("sizeof(arr[0]) = %d\n", sz2);return 0;
}

运行结果:
代码运行结果

  1. 可以看到,当是arr时,结果是48,由于创建的整型数组有12个元素,每个元素大小是4个字节,所以整个数组的大小是48个字节。
  2. 当是arr[0]时,结果是16,由于一行有四个元素,所以是16个字节。
    此时数组名代表的是整个数组,数组的行名代表的是这一行。

二维数组行数和列数的求法:

#include<stdio.h>int main()
{int arr[][4] = { 1,2,3,4,5,6,7,8,9,10,11,12 };//创建二维数组时,行可以省略,列不能省略int row = sizeof(arr) / sizeof(arr[0]);//行数就等于整个数组的大小除以第一行的大小int col = sizeof(arr[0]) / sizeof(arr[0][0]);//列数就等于第一行的大小除以首元素的大小printf("是一个%d行%d列的二维数组\n", row, col);return 0;
}

运行结果:
代码运行结果

在操作符&下数组名的实质

&操作符的作用在上面已经介绍过。
当数组名遇到&时的实质又是什么呢?

#include<stdio.h>int main()
{int arr[3][4] = { {1,2,3},{3,4,5,6},{1,2} };//创建二维数组,三行四列,空白部分自动用0填充printf("&arr[0][0]   = %p\n", &arr[0][0]);printf("&arr[0][0]+1 = %p\n", &arr[0][0] + 1);//打印首地址元素加1printf("\n");printf("&arr   = %p\n", &arr);//打印数组名取地址printf("&arr+1 = %p\n", &arr + 1);//打印数组名取地址加1printf("\n");printf("&arr[0]   = %p\n", &arr[0]);//打印数组首行名取地址printf("&arr[0]+1 = %p\n", &arr[0] + 1);//打印数组首行名取地址加1return 0;
}

运行结果:
代码运行结果
首元素地址加1后多出了4个字节。
数组名取地址加1后多出了48个字节,是所有元素所占空间的大小。
数组首行名取地址加1后多除了16个字节,是首行元素所占空间的大小。
虽然首元素地址和数组名取地址已经数组首行元素取地址的表现相同,但是实质不同。
首元素取地址打印的结果就是首元素的地址,但是数组名取大致代表的整个数组,数组某行元素取地址代表的是这一行的地址。

总结

一般情况下,一维数组的数组名的本质就是首元素的地址,二维数组的数组名是首行元素的地址,只有在两种特殊情况,也就是遇到sizeof操作符和&操作符时,一维数组和二维数组的数组名代表的是整个数组。
二维数组也可以看作是一维数组,这时每一行就是一维数组的一个元素。

这篇关于【C语言学习】数组名的实质的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/221416

相关文章

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Go语言如何判断两张图片的相似度

《Go语言如何判断两张图片的相似度》这篇文章主要为大家详细介绍了Go语言如何中实现判断两张图片的相似度的两种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 在介绍技术细节前,我们先来看看图片对比在哪些场景下可以用得到:图片去重:自动删除重复图片,为存储空间"瘦身"。想象你是一个

Go语言中Recover机制的使用

《Go语言中Recover机制的使用》Go语言的recover机制通过defer函数捕获panic,实现异常恢复与程序稳定性,具有一定的参考价值,感兴趣的可以了解一下... 目录引言Recover 的基本概念基本代码示例简单的 Recover 示例嵌套函数中的 Recover项目场景中的应用Web 服务器中

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

Go语言中使用JWT进行身份验证的几种方式

《Go语言中使用JWT进行身份验证的几种方式》本文主要介绍了Go语言中使用JWT进行身份验证的几种方式,包括dgrijalva/jwt-go、golang-jwt/jwt、lestrrat-go/jw... 目录简介1. github.com/dgrijalva/jwt-go安装:使用示例:解释:2. gi