指针(2)--指针与数组

1.数组名

先看一个案例

b7f513b328164d649d45e640e508f66f.png

在这个代码中,我们提取了数组的首字母的地址和数组名来进行打印,结果是这样的:

acf21938f610431d9cbe0587f49d3ca4.png

可以发现它们两个的地址是相同的,所以我们可以得出

在通常情况下,数组名表示的就是数组首元素(第⼀个元素)的地址。

再来看一个案例:

f064175ecf1941529426b4fbca0e26c7.png

在这个代码中,我们打印sizeof(数组名),结果是这样的:

b7eb43143e2b466eb80dfe5616262140.png

理论上来说如果数组名是数组首元素的地址,那打印出来应该是4或8也就是一个元素所占的字节,但这里却是40。

别急,再看一个案例:

66c0fd02856d4657b700db73fd712e5f.png

这里我们打印数组名和&数组名,也就是取地址数组名,结果是这样的:

c7227c62248b478d832d60c29a8c88eb.png

这里我们不看打印结果,直接看类型,可见一个是int[10]一个是int[10]* 也就说一个是数组一个是数组解引用。

所以除了通常情况下,还有两个例外:

• sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表示整个数组,计算的是整个数组的大小,单位是字节

• &数组名,这里的数组名表示整个数组,取出的是整个数组的地址(整个数组的地址和数组首元素的地址是有区别的)

我们还可以通过另一个代码来更好理解第二个例外:

ee064d67ab324f7e9be945cc5f694ab7.png

6f3a8e8c9afc4a228d61f69a9636984e.png

我们发现:

&arr[0]和&arr[0]+1相差4个字节,arr和arr+1 相差4个字节,是因为&arr[0] 和 arr 都是首元素的地址,+1就是跳过⼀个元素。

但是&arr 和 &arr+1相差40个字节,这就是因为&arr是数组的地址,+1 操作是跳过整个数组的。

2.使用指针访问数组

4876c470e9a34bd0a7a70f0d50766fc0.png

既然arr是数组的首元素的地址,那么它赋值给p时,其实可以认为p等价于arr。所以arr[i]也就等价于p[i]。

2614a0d12ab44bdaa8c26389b90b7c0b.png

我们需要知道的一个很重要的事实是:

数组其实就是指针,它的底层含义就是地址。

从1中我们也得知数组名就是其首元素地址。

所以其实满足这样的关系式:
arr[i]==*(arr+i)==*(i+arr)==i[arr](满足交换律)

数组元素的访问在编译器处理的时候,也是转换成首元素的地址+偏移量求出元素的地址,然后解引用来访问的。

3.一维数组传参本质

b88652d3aa544ad98d7c5ffe97e6080e.png

44f60f7f67b448419f7ba13c292859a7.png数组传参本质上传的是数组首元素的地址。sz1是整个数组元素的个数(因为它是直接打印主函数中的数组元素个数);

而sz2是函数中的数组元素的个数,但这里其实取的只是首元素,也就只有一个了。

函数形参的部分是使用指针变量来接收首元素的地址。那么在函数内部我们写 sizeof(arr) 计算的是⼀个地址的大小(单位字节)而不是数组的大小(单位字节)。

正是因为函数的参数部分是本质是指针,所以在函数内部是没办法求数组元素个数的。

4.二级指针/多级指针

指向某一个变量的地址就是指针,当这个变量本身不是指针时,那么这里的指针叫做一级指针。

我们知道,指针变量是指向某一类型数据的内存地址的变量,它作为变量的本质是不会改变的。

那么只要是变量就会有地址。

所以指针变量的地址就叫做二级指针。它是指向指针的指针。

4c0b77e42be04bd9b511b9c5af7db8c9.png

那么如果我们要通过pp来找到或者改变a变量,就要解引用两次。

6d9e3523935b4885914dd7c81498f4c3.png

既然二级指针是指向指针的指针,那么肯定就有指向二级指针的指针,被称为三级指针,还有四级指针五级指针...它们都被统称为多级指针。

5.指针数组

这个名词的主体是数组,所以顾名思义:

指针数组就是数组内所有元素都是指针的数组,也就是用来存放指针的数组。        

427f418d36bf42a493cd75cadcb070c3.pngab5484bd2dd944d3b6f47a4aaa90b023.png

对比两个数组,它们的区别在于一个数组内的元素类型是整型int,另一个是整形指针int*。

既然指针数组的每一个元素都是地址,那么元素的数量也就是数组内指向的地址的数量。

6.指针数组与数组指针

指针数组是元素是指针的一类数组,它的本质是数组。

数组指针是指向某个数组的指针,它的本质是指针。

7.指针数组与普通数组的联系

1. 数据类型不同

普通数组的数据类型是普通的类型,但指针数组的数据类型实际上是普通的类型再加上*号,用来表明该数据是指针。

2. 存储的内容不同

普通数组直接存储数据的值,而指针数组存储的是指针,即存储了数据的内存地址。

3. 访问方式

普通数组可以直接通过下标访问数组中的元素,而指针数组需要通过指针来访问数组中的元素,即先获取指针,然后通过指针访问数据。

在这里我们可以思考:是否可以用指针数组来模拟实现二维数组呢?毕竟只需要获取指针就可以访问数据,也就可以模拟实现二维数组了。

23b577c976234b85a51ac0ed255cb2b7.png

注意:上述的代码模拟出二维数组的效果,实际上并非完全是⼆维数组,因为每⼀行并非是连续的。

4. 灵活性

指针数组的元素可以指向不同类型的数据,而普通数组的元素必须是相同类型的数据。

总而言之,指针数组它也是一个数组,只不过有它特殊的用法。

  • 31
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

'Skrrapper

谢谢你的支持!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值