原文地址:http://blog.csdn.net/czmpersist/article/details/9288205
上一回合讲了指针内存与地址,最后提到了类型说明,这一回合继续指针类型的说明。
类型说明
要了解指针,多多少少会出现一些比较复杂的类型,今天先介绍一下如何完全理解一个复杂类型,要理解复杂类型是有规律可寻的,找到规律后一切变得很简单,一个类型里会出现很多运算符,他们也像普通的表达式一样,有优先级,其优先级和运算优先级一样,所以我总结了一下其原则:
从变量名处起,根据运算符优先级结合,一步一步分析。
下面我将用到的几个操作符表示出来,方便大家参考,从上到下优先级是逐渐降低的。
Associativity and Operator 操作符及其结合性 |
Function 功能 |
Use 用法 |
|
L |
[] |
subscript(下标) |
variable [ expr ] |
L |
() |
function call(函数调用) |
name (expr_list) |
R |
* |
dereference(解引用) |
*expr |
下面让我们先从简单的类型开始慢慢分析吧:
Ø int p;
这是一个普通的整型变量。
Ø int *p;
首先从 P 处开始,先与*结合,所以说明P是一个指针,然后再与int结合,说明指针所指向的内容的类型为 int 型,所以P是一个返回整型数据的指针。
Ø int p[3];
首先从 P 处开始,先与[]结合,说明 P 是一个数组,然后与 int 结合,说明数组里的元素是整型的,所以P是一个由整型数据组成的数组。
Ø int *p[3];
首先从 P 处开始,先与[]结合,因为[]优先级比*高,所以P是一个数组,然后再与*结合,说明数组里的元素是指针类型,然后再与int结合,说明指针所指向的内容的类型是整型的,所以P是一个由返回整型数据的指针所组成的数组。
Ø int (*p)[3];
首先从P处开始,先与*结合,说明P是一个指针然后再与[]结合(与"()"这步可以忽略,只是为了改变优先级),说明指针所指向的内容是一个数组,然后再与int结合,说明数组里的元素是整型的。所以 P 是一个指向由整型数据组成的数组的指针。
Ø int **p;
首先从P开始,先与右*结合(*结合方向从右至左),说是 P 是一个指针,然后再与*结合,说明指针所指向的元素是指针,然后再与int结合,说明该指针所指向的元素是整型数据。由于二级指针以及更高级的指针极少用在复杂的类型中,所以后面更复杂的类型我们就不考虑多级指针了,最多只考虑一级指针。
Ø int p(int);
从P处起,先与()结合,说明P是一个函数,然后进入 ()里分析,说明该函数有一个整型变量的参数然后再与外面的int结合,说明函数的返回值是一个整型数据。
Ø int (*p)(int);
从 P 处开始,先与指针结合,说明P是一个指针,然后与()结合,说明指针指向的是一个函数,然后再与()里的int结合,说明函数有一个int型的参数,再与最外层的int结合,说明函数的返回类型是整型,所以P是一个指向有一个整型参数且返回类型为整型的函数的指针。
Ø int *(*p(int))[3];
从 P 开始,先与()结合,说明P是一个函数,然后进入()里面,与int结合,说明函数有一个整型变量参数,然后再与外面的*结合,说明函数返回的是一个指针,然后到最外面一层,先与[]结合,说明返回的指针指向的是一个数组,然后再与*结合,说明数组里的元素是指针,然后再与int结合,说明指针指向的内容是整型数据。所以P是一个参数为一个整数据且返回一个指向由整型指针变量组成的数组的指针变量的函数。
上面这些是参考谭浩强版C语言相关说明,以及在网上查找到的资料综合一下的结果,帮助大家好好理解指针类型。如果上面的所以情况都能理解,对于指针定义就 差不多,太多太复杂的定义在实际的开发过程中很难懂,可读性较低,应用也就不会太广泛,望读者自己衡量。注意:以上所有定义以后的指针并没有初始化,也就 是说指针所指的位置我们无法知道,这也是编程人员必须深刻认识的一点。
上面所讲的只是针对int类型,对于char,double,结构体等类型原理一样,望大家知一反三。好了,今天就讲到这里,一步一步来,不要着急,当你基础的把握好了就可以变化,活学活用了,还是一样,大家如果有的建议,或者没有涉及到的,往大家指点,小弟将积极采纳!