关于我、重生到500年前凭借C语言改变世界科技vlog.12——深入理解指针(2)
数组名与地址
有这么一个数组,数组名为 arr
1 | int arr[10] = {1,2,3,4,5,6,7,8,9} |
在这里 &arr[0] 指的是首元素地址,其实本质上数组名就代表的是数组首元素地址
接下来我们将对以下几个名字进行区分
1.arr
2.sizeof(arr)
3.&arr
arr
1 |
|
根据以上保留取地址符&和只用数组名,打印对应的地址可以发现
两个的地址都相同,这就说明了 &arr[0] = arr,即数组名就是首元素地址
sizeof(arr)
1 |
|
这里关键字 sizeof 计算 arr 的大小,你可能会觉得应该输出首元素的大小?
不,这里的 arr 不代表首元素大小,而是表示整个数组,计算整个数组的大小,单位是字节
&arr
1 |
|
打印 &arr 和 &arr+1 地址,假设他是跳过一个字节的地址
很显然,根据代码结果,两段代码地址相差40,那么假设就是错的,所以 &arr 表示的是整个数组的大小
总结:只有 &arr 和 sizeof(arr) 有特殊含义,其他情况下的数组名都代表数组首元素地址
指针访问数组
1 |
|
使用指针访问数组在一些环境下,会更接近本质,更加简洁易懂
将 *(p+i) 换成 p[i] 也是能够正常打印的
在这里解引用操作符和下标引用符产生的效果相同
本质上p[i] 是等价于 *(p+i),数组元素的访问在编译器处理的时候,也是转换成首元素的地址+偏移
量求出元素的地址,然后解引用来访问的
一维数组传参本质
在前面学习的函数篇章中,数组可以传递给自定义函数,那么数组传递的本质是什么呢?
1 |
|
通过以上代码调试后可以发现,sz1 = 10,sz2 = 1,同样的算法应该 sz1 = sz2 =10 ,但穿过去的形参根据结果 sizeof(arr) 只计算出一个元素的大小,前面我们提到 arr 表示首元素的地址,那么在数组传参的时候,传递的是数组名,也就是数组传参的时候传递的是首元素的地址
所以函数的形参应该用指针变量来接收一个地址,这也证实了数组传参的本质是传递首元素地址
形参部分既可以写成 int * arr 的指针形式,也可以写成 arr[ ] 的数组形式
指针数组
什么是数组指针?什么是指针数组?这里我们重点介绍一下指针数组
指针数组是一个数组,比如整型数组存放的数组,字符数组存放的是字符,那么指针数组存放的就是指针
int* arr[5] = 5个int*
每一个数组里的指针都指向一个地址
二级指针
正常的一个常见的指针是一级指针,那么在一个指针里面再嵌套一个指针就是二级指针
是变量就有地址,指针变量也不例外,比如把指针变量的地址需要存放在另一个指针变量里
1 |
|
*pii 通过对 pii 中的地址进行解引用,这样找到的是 pi , *pii 其实访问的就是 pi
**pii 先通过 *pii 找到 pi ,然后对 pi 进行解引用操作: *pi ,那找到的是 i
博主最近在准备蓝桥杯的校内比赛,更新可能会有所减缓,大家多多谅解!:)