指针定义


    1. int p;

    这是一个普通的整型变量


    1. int* p;

    首先从P 处开始,先与*结合,所以说明P 是一个指针,然后再与int 结合,说明指针所指向的内容的类型为int 型.所以P是一个返回整型数据的指针


    1. int p[3];

    首先从P 处开始,先与[]结合,说明P 是一个数组,然后与int 结合,说明数组里的元素是整型的,所以P 是一个由整型数据组成的数组


    1. int* p[3];

    首先从P 处开始,先与[]结合,因为其优先级比高,所以P 是一个数组,然后再与结合,说明数组里的元素是指针类型,然后再与int 结合,说明指针所指向的内容的类型是整型的,所以P 是一个由返回整型数据的指针所组成的数组


    1. int(*p)[3];

    首先从P 处开始,先与*结合,说明P 是一个指针然后再与[]结合(与”()”这步可以忽略,只是为了改变优先级),说明指针所指向的内容是一个数组,然后再与int 结合,说明数组里的元素是整型的.所以P 是一个指向由整型数据组成的数组的指针


    1. int** p;

    首先从P 开始,先与结合,说是P 是一个指针,然后再与结合,说明指针所指向的元素是指针,然后再与int 结合,说明该指针所指向的元素是整型数据.由于二级指针以及更高级的指针极少用在复杂的类型中,所以后面更复杂的类型我们就不考虑多级指针了,最多只考虑一级指针.


    1. int p(int);

    从P 处起,先与()结合,说明P 是一个函数,然后进入()里分析,说明该函数有一个整型变量的参数,然后再与外面的int 结合,说明函数的返回值是一个整型数据


    1. Int(*p)(int);

    从P 处开始,先与指针结合,说明P 是一个指针,然后与()结合,说明指针指向的是一个函数,然后再与()里的int 结合,说明函数有一个int 型的参数,再与最外层的int 结合,说明函数的返回类型是整型,所以P 是一个指向有一个整型参数且返回类型为整型的函数的指针


    1. int* (*p(int))[3];

    可以先跳过,不看这个类型,过于复杂从P 开始,先与()结合,说明P 是一个函数,然后进入()里面,与int 结合,说明函数有一个整型变量参数,然后再与外面的结合,说明函数返回的是一个指针,,然后到最外面一层,先与[]结合,说明返回的指针指向的是一个数组,然后再与结合,说明数组里的元素是指针,然后再与int 结合,说明指针指向的内容是整型数据.所以P 是一个参数为一个整数据且返回一个指向由整型指针变量组成的数组的指针变量的函数.

    指针指向

    指针作为一种普通的数据结构,有自己的地址,也存放着值,但是指针的值是一个地址值,可以把地址赋给指针。
    下图中,指针p存放的是arr的地址,而arr作为数组来讲,arr的第一个元素的地址也就是arr的首地址;
    那么访问p的地址就访问了arr的地址,p的地址值加上4(这个4也就是sizeof int)就访问到了arr的下一个元素。
    指针.svg

    指向指针的指针

    代表这个指针可以指向一个指针,因为指针也有地址,这个地址可以被指向,同理*也成立。

    1. #include <iostream>
    2. using namespace std;
    3. int main() {
    4. int i = 0;
    5. int *p = &i;
    6. int **pp = &p;
    7. int ***ppp = &pp;
    8. cout << ***ppp;
    9. }

    但是上一级指针只能指向相邻的下一级指针,不能进行跨级指向。
    例如*ppp = &p就是错误的。