当前位置: 首页 > news >正文

指针的介绍2前

1.数组名的理解

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

观察得到,数组名就是数组首元素的地址

但有2个例外:

1.

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

如果在这段代码中,arr还是数组首元素的地址的话

那么打印结果就应该是48

但打印结果是36

注意到数组一共有 9 个 int型 的元素

数组的大小正好是36个字节

所以:sizeof(数组名)中,数组名代表的是整个数组

2.

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

 

上面的代码示意图如下:

 &arr[0] 、arr、&arr 拿到的地址相同

进行+1操作之后,

&arr[0] 、arr 都跳过4个字节,即跳过一个 int 型d的数组元素

说明它俩的指针类型相同

&arr 跳过 36 个字节,即跳过了一个数组

所以:&数组名 中, 数组名代表的是整个数组

总结:

(1)sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩, 单位是字节

(2)&数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素 的地址是有区别的)

(3)除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址

2.[]下标引用操作符

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9 };int sz = sizeof(arr) / sizeof(arr[0]);for (int i = 0; i < sz; ++i)printf("%d", arr[i]);return 0;
}

打印数组元素时,使用了 [] 操作符

实际上,arr[i] 的本质是 *(arr + i)

所以就是用数组首元素的地址的移动来遍历数组

(arr + i) 与 (i + arr) 一致

所以

arr[i] == *(arr + i) == *(i + arr) == i[arr]

上述等式验证如下:

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

总结:

arr[i] 的本质是 *(arr + i)

数组元素的访问在编译器处理的时候,

是转换成⾸元素的地址+偏移量求出元素的地址,

然后解引⽤来访问的。

 3.一维数组传参的本质

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>void test(int arr[])
{int sz2 = sizeof(arr) / sizeof(arr[0]);printf("%d\n", sz2);
}int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9 };int sz1 = sizeof(arr) / sizeof(arr[0]);printf("%d\n", sz1);test(arr);return 0;
}

x86平台运行下:

x64环境下:

main函数中,sz1,毋庸置疑,是数组元素的个数

test 函数中, 我的初心也是求传入数组的元素个数

可不管是x64还是x86环境下,结果都不正确

仔细分析:

sz2 = sizeof(arr) / sizeof(arr[0])

sizeof(arr[0]) == 4

x86环境下

sz2 == 1,所以sizeof(arr) == 4

如果arr代表的是整个数组,那么sizeof(arr) == 36

显然arr代表的是数组首元素的地址,此时sizeof(arr) == 4

x64环境下

sz2 == 2,所以sizeof(arr) == 8

如果arr代表的是整个数组,那么sizeof(arr) == 36

显然arr代表的是数组首元素的地址,此时sizeof(arr) == 8

总结:

本质上,数组传参传递的是数组⾸元素的地址

http://www.lryc.cn/news/527746.html

相关文章:

  • 16.Word:石油化工设备技术❗【28】
  • Python-基础环境(01) 虚拟环境,Python 基础环境之虚拟环境,一篇文章助你完全搞懂!
  • Dest1ny漏洞库:用友 U8-CRM 系统 ajaxgetborrowdata.php 存在 SQL 注入漏洞
  • java.sql.Date 弃用分析与替代方案
  • HarmonyOS:状态管理最佳实践
  • 如何提高新产品研发效率
  • MongoDB平替数据库对比
  • JavaScript系列(46)-- WebGL图形编程详解
  • YOLO目标检测4
  • 十三先天记
  • 【论文阅读笔记】“万字”关于深度学习的图像和视频阴影检测、去除和生成的综述笔记 | 2024.9.3
  • Android AOP:aspectjx
  • 前端【11】HTML+CSS+jQUery实战项目--实现一个简单的todolist
  • 2025课题推荐——USBL与DVL数据融合的实时定位系统
  • 滑动窗口详解:解决无重复字符的最长子串问题
  • 第05章 11 动量剖面可视化代码一则
  • MySQL的复制
  • Cpp::IO流(37)
  • 基于OpenCV实现的答题卡自动判卷系统
  • 如何将电脑桌面默认的C盘设置到D盘?详细操作步骤!
  • 二十三种设计模式-享元模式
  • 算法【有依赖的背包】
  • A7. Jenkins Pipeline自动化构建过程,可灵活配置多项目、多模块服务实战
  • 飞牛NAS新增虚拟机功能,如果使用虚拟机网卡直通安装ikuai软路由(如何解决OVS网桥绑定失败以及打开ovs后无法访问飞牛nas等问题)
  • 蓝桥杯例题四
  • 八股——Java基础(四)
  • CVE-2023-38831 漏洞复现:win10 压缩包挂马攻击剖析
  • 【自然语言处理(NLP)】深度循环神经网络(Deep Recurrent Neural Network,DRNN)原理和实现
  • Linux 命令之技巧(Tips for Linux Commands)
  • 【文星索引】搜索引擎项目测试报告