【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

相关文章

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

C语言中位操作的实际应用举例

《C语言中位操作的实际应用举例》:本文主要介绍C语言中位操作的实际应用,总结了位操作的使用场景,并指出了需要注意的问题,如可读性、平台依赖性和溢出风险,文中通过代码介绍的非常详细,需要的朋友可以参... 目录1. 嵌入式系统与硬件寄存器操作2. 网络协议解析3. 图像处理与颜色编码4. 高效处理布尔标志集合

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服

C 语言中enum枚举的定义和使用小结

《C语言中enum枚举的定义和使用小结》在C语言里,enum(枚举)是一种用户自定义的数据类型,它能够让你创建一组具名的整数常量,下面我会从定义、使用、特性等方面详细介绍enum,感兴趣的朋友一起看... 目录1、引言2、基本定义3、定义枚举变量4、自定义枚举常量的值5、枚举与switch语句结合使用6、枚

Go 语言中的select语句详解及工作原理

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语... 目录Go 语言中的 select 是做什么的基本功能语法工作原理示例示例 1:监听多个通道示例 2:带

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

C语言中的数据类型强制转换

《C语言中的数据类型强制转换》:本文主要介绍C语言中的数据类型强制转换方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C语言数据类型强制转换自动转换强制转换类型总结C语言数据类型强制转换强制类型转换:是通过类型转换运算来实现的,主要的数据类型转换分为自动转换

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时

使用C语言实现交换整数的奇数位和偶数位

《使用C语言实现交换整数的奇数位和偶数位》在C语言中,要交换一个整数的二进制位中的奇数位和偶数位,重点需要理解位操作,当我们谈论二进制位的奇数位和偶数位时,我们是指从右到左数的位置,本文给大家介绍了使... 目录一、问题描述二、解决思路三、函数实现四、宏实现五、总结一、问题描述使用C语言代码实现:将一个整