题目:
最近看到一道模拟题,题目如下:
#include<stdio.h>
#include<iostream>
#include<vector>
#include<algorithm>
#include<string>
using namespace std;
int main(){
int n[][3] = { 10, 20, 30, 40, 50, 60 };
int(*p)[3];
p = n;
cout << p[0][0] << "," << *(p[0] + 1) << "," << (*p)[2] << endl;
cout << (*p)[3] << endl;
cout << (*p + 1)[1] << endl;
return 0;
}
先思考一下,然后断点设置后结果如下:
分析:
首先,我们来看这一句:
int n[][3] = { 10, 20, 30, 40, 50, 60 };
它代表的是下图,子数组是长度为3的数组,如下图:(其中黑色的部分)
而
int(*p)[3];
p = n;
表示p是一个指针,并且指向3个int元素的数组,在这里它指向了n所表示的数组。
那么(*p)就是指向了n[0]所代表的三个元素,即上图中蓝色部分
*(p + 1)中,表示指向下一个位置(下一个三个数组)所对应的值,即上图中绿色部分
(*p + 1)或(*p) + 1意义一样(因为*的优先级>加号的优先级),表示(*p)的下一个元素,即指向20的指针,注意这里不是20,图中橙色部分。
注意对p加1和对(*p)加1的不同!其中加的是三个int大小,后者因为已经解引用过了,加的一个int大小。
逐个来分析:
(*p)对应的是指向{10,20,30}的指针
*(p + 1)对应的是指向{40,50,60}的指针
(*p + 1)对应的是指向20的指针
*(p[1] + 1)中p[1]就相当于*(p + 1),指向了{40,50,60},再加1表示加一个int大小,最后解引用了,所以对应50。
p[0][0]不用说,对应10。
*(p[0] + 1),p[0]对应于(*p),是指向{10,20,30}的指针,再加1表示加一个int大小,最后解引用了,所以对应20。
(*p)[2]中(*p)表示指向{10,20,30}的指针,再对其[2],就是对应30。
(*p)[4]中4越界了,那么跳到了第二行,对应的是50。
(*p + 1)[1]中(*p + 1)对应指向20的指针,对其[1]相当于向后一个int的位置并解引用,相当于*((*p + 1) + 1),所以对应30。(x[1]相当于是*(x + 1))
OK,叙述完毕,多谢小双莅临指导。
——Apie陈小旭