目录
一、字符指针
1.1字符指针的另一种用法
在初学指针时,我们了解了如何将一个字符放入字符指针中,例如:
然而,我们对于字符指针也有另一种使用方法。我们可以将字符串也放入字符指针中,例如:
这样一来我们便将abcdef这串字符放入了指针p中。在解析这段代码时,大家经常出现一个误区,那便是我们将这段字符串整体放入了指针p中,这实际上是错误的认知。在这串代码中,我们仅将字符串首个字符的地址放入了指针p中。
1.2 一道有意思的面试题
经过验证我们发现
那么为什么会出现这种情况呢?
对于str1与str2而言,他们存储的是常量字符串,分配到的内存空间是不同的,若比较两个数组的地址,当然是不同的。
而在程序中,如果不同指针的内容是一样的,那么这些指针的地址都是相同的。所以我们比较str3与str4时,他们的地址相同。
二、数组指针
2.1数组指针是数组还是指针?
数组指针实际上是指针。
那么我们现在来创建一个数组指针
在上述代码中,pa就是一个数组指针,它指向arr数组,存储arr数组的首地址。
2.2 数组名与&数组名的区别
我们曾经学习过,数组名就是数组的首地址,那么为何在创建数组指针时,要使用&呢?
通过分别打印它们的地址,我们发现它们的地址居然是相同的。
那么我们分别将它们+1,出现了如下的结果。
通过计算我们可知,对于arr而言,加1后指向的地址+4个字节,跳过一个整型的内存空间。二对于&arr而言,加1后指向的地址+12,跳过了3个整型内存空间,正好是我们创建的一个数组的空间。由此我们可以得知,&arr+1代表着跳过一整个数组空间,这便是arr与&arr在计算时的区别。
简而言之,数组名为数组首元素的地址,而&数组名为数组的地址,二者虽然在数值上相同,但代表的意义完全不同。
2.3 数组指针的应用
在函数中,我们通过使用数组指针,基于二维数组的行,进行操作,使得i每+1,指针便跳转至二维数组下一行,从而实现分行打印二维数组。在这里我们需要知道,在二维数组中,数组名代表着二维数组的第一行,所以我们可以通过对于行+1来进行访问。
三、数组参数、指针参数
3.1 一维数组传参
让我们来判断以下两个函数的传参是否正确。
函数1:
通过我们已学的知识,可以轻松判断出函数1的前两种传参方式是正确的。我们已知一维数组的数组名,为数组首元素地址,那么便可以对其数组名进行解引用,从而获得数组名的地址,即为数组首元素地址,故而第三种传参方式也是正确的。
函数2:
函数2 的第一种传参方式即为我们此次所介绍的指针数组传参。对于第二种传参方式,我们需要分析arr2的组成,arr2为指针数组,所以arr2的元素为指针。而数组名代表着首元素的地址,即为指针的指针,所以我们可以对于arr2的数组名进行解引用,从而获得arr2数组的首元素地址。所以第二种传参方式也是正确的。
3.2 二维数组传参
寻常的二维数组传参方式,我们便不再赘述,这里我们来判断一下通过指针进行二维数组传参的方式。
第一种传参方式是错误的。我们已知二维数组的数组名为第一行元素的地址,将其解引用得到的为指针的指针,而我们需要的是第一行的地址,所以是错误的。
第二种传参方式也是错误的。第二种传参方式所表述的是将指针数组作为参数,更是大错特错。
第三种传参方式是正确的。加上括号之后表示为数组指针,含有五个元素,恰好符合二维数组的列数,又由于数组指针本质是指针,所以可以用来作为参数。
第四种传参方式是错误的。二级指针用来接收一级指针的地址,不能作为二维数组的参数使用。
3.3 一级、二级指针传参
一级指针传参只需要使用一级指针接受即可。
二级指针传参时,也要使用二级指针来接收。
当函数的参数为二级指针的时候,可以接收什么参数?
以上三种传参方式都是正确的。第三种传参方式是将指针数组的数组名作为参数,由于指针数组的各个元素都是指针,所以数组名作为首元素地址,便是第一个指针元素的地址。
四、函数指针
数组名即为指针,那么函数是否也存在指针呢?我们通过如下一段代码来验证一下
通过对于Add函数取地址并打印后,我们发现函数也是存在地址。
那么我们如何来存放函数的地址呢?寻常的创建指针变量来存放函数地址的方法是行不通的,我们需要通过指定格式来存放函数指针。
例如:
第一个int代表函数返回类型,(*pf)即为创建的函数指针变量,第二个括号中为函数的参数。通过这种格式,我们便可以存储函数指针。
其实我们也可以将&Add改为add,不加&符号也是可以达到同样效果。这是由于在函数中,&函数名与直接使用函数名是相同的,都代表函数的地址。
例如: