在初次学习指针中,我们了解到以下内容:
- 指针就是一个变量,用来存放地址,地址唯一标识一块空间
- 指针的大小是固定4/8个字节(32/64位平台)
- 指针是有类型的,指针的类型决定了指针的 +- 整数的步长和指针解引用操作的时候的权限(访问几个字节)。
- 指针的相关运算 注:本次主要讨论指针更高级的用法
1.字符指针
一般使用:
还有一种会使用方式:
代码 char* pstr = "hello bit."
; 特别容易让同学以为是把字符串 hello bit 放到字符指针 pstr 里了,但是本质是把字符串 hello bit. 首字符的地址放到了pstr中
如图所示:
上面代码的意思是把一个常量字符串的首字符 h 的地址存放到指针变量 pstr 中。
那可能会有这样的问题:
最终输出结果为:
这里str3和str4指向的是同一个字符串常量.C/C++会把常量字符串的存储到单独的一个内存区域,当几个指针指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4不同。
2.指针数组
例如:
int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5];//二级字符指针的数组
注意:指针数组是一个数组,里面存放的是一个一个的指针。
3.数组指针
例如:
int (*p)[10];
//解释:p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
//这里要注意:[]的优先级要高于*号的,所以必须加上()来保证p先和*结合。
注意:数组指针是一个指针,指向的是一个数组
&数组名 和 数组名
#include <stdio.h>
int main()
{
int arr[10] = { 0 };
printf("arr = %p\n", arr);
printf("&arr= %p\n", &arr);
printf("arr+1 = %p\n", arr+1);
printf("&arr+1= %p\n", &arr+1);
return 0;
}
通过以上代码的验证及结果可以发现:
- &arr和arr的值是一样的
- &arr取出的是整个数组的地址,我们可以从它+1后的改变量看出
- arr是数组首元素的地址,在这个题里就是int*类型
- 数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40
4.函数指针
首先看一段代码
#include <stdio.h>
void test()
{
printf("hehe\n");
}
int main()
{
printf("%p\n", test);
printf("%p\n", &test);
return 0;
}
运行结果:
输出的是两个地址,这两个地址就是 test 函数的地址。
要保存函数的地址,我们就引出了函数指针,例如存放test函数的地址的函数指针void (*fptr)();
理解:fptr先和*结合,说明fptr是指针,指针指向的是一个函数,指向的函数无参数,返回值类型为void。
5.函数指针数组
听其名便可知道只是一个数祖,里面存放的是一个个的函数指针,但是如何定义呢?
int (*parr1[10]])();
int *parr2[10]();
int (*)() parr3[10];
根据理解我们可以看到int (*parr1[10]])();
是正确的,可以这样理解:parr1 先和 [] 结合(因为[]结合性比优先级高),说明parr1是数组,数组的内容是什么呢? 是 int ()() 类型的函数指针.这样我们就定义了一个函数指针数组.
6.指向函数指针数组的指针
理解:指向函数指针数组的指针是一个 指针, 指针指向一个 数组 ,数组的元素都是 函数指针。这样我们就可以定义出来了 看如下代码:
void test(const char* str)
{
printf("%s\n", str);
}
int main()
{
//函数指针pfun
void (*pfun)(const char*) = test;
//函数指针的数组pfunArr
void (*pfunArr[5])(const char* str);
pfunArr[0] = test;
//指向函数指针数组pfunArr的指针ppfunArr
void (*(*ppfunArr)[10])(const char*) = &pfunArr;
return 0;
}
我们就介绍到这里,希望对小伙伴们的学习有所帮助,若本章节中有错误,希望大家及时提出,大家一起探讨。