学习C的过程中,感觉指针是比较头疼的一个点,而且它还与很多其他的点有关,比如数组,字符串,函数,结构,所以在此做个总结。
C指针的定义:
C中的指针到底是什么,其实它就是一种特殊的数据类型,可以把它和基本数据类型的声明做个类比:
int a = 8; //声明了一个变量a,它存储的值是int类型的整数8
char ch = 'a'; //声明了一个变量ch,它存储的值是char类型的字符a
int *p=&a; //声明了一个变量p,它存储的值是一个内存地址,例如:008FFE00,而这个地址存储的值类型又是int类型的数据
所以指针就是:一种用来保存内存地址的数据类型
C指针相关术语
指针指向类型:指针保存的数据是地址,这个地址代指内存中某一块区域,这一块区域也会保存数据,这个地址保存的数据类型就是指针指向类型
C复杂指针解析
指针简单的声明还好理解,但如果不是很熟的话,复杂的指针声明就歇菜了,比如下面的:
int *p; //p是一个指针
int p[3]; //p是一个数组
int *p[3]; //p是一个数组
int (*p)[3]; //p是一个指针
int(*p)(int); //p是一个指针
int(*(*p())[])(); //p是一个函数
从第三个开始,是不是就懵了,反正在我没搞清楚复杂声明的时候,从第三个开始就懵逼了
下面解析上面的声明,第一个和第二个就不说了,从第三个开始:
int *p[3];
1,首先从变量名p开始,因为[ ]的优先级比星号 * 高,所以p先与[ ]结合组成一个数组,变量p就是一个数组了,这个数组有3个元素,元素类型未知
2,把p[3]
当做一个整体,再结合左边的*
,得知数组元素类型为指针类型,指针的指向类型未知
3,把*p[3]
当一个整体,结合左边的int
,得知这个指针指向的是int类型的数据
4,所以,p是一个有3个元素的数组,数组元素类型为指向了int类型数据的指针
int (*p)[3];
1,从变量名p开始,
p
和*
被括号包裹,所以优先组合括号里面,*
和p
组合成一个指针,所以p
是一个指针,指向类型未知
2,再看括号外面,把(*p)
当做一个整体,右边是[3]
,所以指针p指向了一个数组,数据元素类型未知
3,把(*p)[3]
当做一个整体,左边是int
,所以这个数组的元素类型是int
4,所以,p是一个指向了有3个int型元素的数组的指针
int(*p)(int);
1,从p开始,
*
和p
被括号包裹,优先运算组合成一个指针,指针指向类型未知
2,然后再看右边(int)
,指针p指向了一个参数类型为int的函数,函数返回值类型未知
3,把(*p)(int)
当一个整体,看左边int
,得知这个函数的返回值类型为int
4,所以,p是一个指向了参数类型为int,返回值类型为int的函数的指针
int(*(*p())[])();
1,从p开始,p右边有个括号,所以p还是一个函数,函数返回值类型未知
2,把p()
当一个整体,看左边*
,得知这个函数的返回值类型是一个指针,指针指向类型未知
3,把(*p())
当一个整体,右边的[ ]
比左边的*
优先级高,所以优先与[ ]
组合,得知指针指向的是一个数组,数组元素类型未知
4,把(*p())[]
当一个整体,看左边*
得知,数组元素类型是一个指针
5,把(*(*p())[])
当一个整体,看右边()
得知,数组元素的指针指向的是一个函数
6,最左边的int说明这个函数的返回值是int类型
7,所以,p是一个返回值为指向了一个数组的指针类型的函数,这个数组的元素类型为指向了一个返回值为int型的函数的指针