最近在重新学习一遍C++的内容,由于C++的指针部分比较复杂,而且很抽象所以大学的时候学过的东西现在也忘了好多。这次回头再看,感觉对指针部分理解更深了,也越来越觉得指针可爱了。
记录一个指针的题目:
int arr[6][2] = {1,2,3,4,5,6,7,8,9};
int (*p)[2] = arr;cout << "**p ==" << **p <<endl;
cout <<" **p++ == "<< **p++ <<endl;
cout <<" **++p== "<< **++p <<endl;
答案是:放在最后面吧~
下面分析这段代码
1、int arr[6][2] = {1,2,3,4,5,6,7,8,9};
第一句代码定义了一个名字叫arr的二维数组,存放的内容都是int类型,大小是6*2=12,但是只放了9个元素。数组申请的内存空间没用到的默认填0。具体看下这个二维数组,6行,2列,相当于有6个元素的一维数组{a,b,c,d,e,f,g},其中每个元素都是一个2个元素的一维数组{i,j}, 即{{1,2},{3,4},{5,6},{7,8},{9,*},{*,*}} *表示在内存中是0。
然后我打印了这个数组在内存中的内容如下图1:
可以看出,不论几维数组,在内存中存放都是按照一维的顺序存放的。起始地址是0x7FFF5FBFF7C0,每个int类型的值占用4个字节,所以很明显能看到数据的存放。
在此处可以列出arr数组中每个元素的地址:
cout << arr[0] << endl;//0x7fff5fbff7c0
cout << arr[1] << endl;//0x7fff5fbff7c8
cout << arr[2] << endl;//0x7fff5fbff7d0
cout << arr[3] << endl;//0x7fff5fbff7d8
cout << arr[4] << endl;//0x7fff5fbff7e0
cout << arr[5] << endl;//0x7fff5fbff7e8
cout << arr[6] << endl;//0x7fff5fbff7f0
2、int (*p)[2] = arr;
定义一个名字为p的一个指向二维数组的指针,也就是数组指针,这里要区别与这种写法:int *q[2] 表示q是一个指针数组,存两个地址(指针)元素。而p存放的是一个地址,这个地址指向一个大小为2的数组。所以在此处把arr看成{{1,2},{3,4},{5,6},{7,8},{9,*},{*,*}}样式比较容易理解。p指向的是arr的首地址,也可以写成int(*p)[2] = &arr[0];表示取arr第0个元素的地址,然后让p指向该地址。
在内存中看如下图2
左侧红框是指针p在内存中的位置,可以看出地址是0x7FFF5FBFF708。右侧红框中是p中存放的内容:00007FFF5FBFF7C0,很明显这也是一个地址,该地址正是arr数组的起始地址。
3、cout << "**p ==" << **p <<endl;
cout <<" **p++ == "<< **p++ <<endl;
cout <<" **++p== "<< **++p <<endl;
是要打印指针p指向内容的值,然后对指针进行偏移多次取值,因为p指向的内容也是一个地址,所以用**p来取最后的值。
“**p” 很明显**p = 1,所以第一个打印的值是1。
“**p++” 表示先取值,后计算,也就是先取出**p的值,然后进行指针偏移一位(此处是一位是指一个**p的占用的内存,也就是8位),所以在打印这句代码的结果的时候**p指向的内容还是不变的,因此第二个值打印的也是1。但是在打印下一句代码之前p已经进行偏移了。
“**++p” 表示先计算,再取值。因为在第二句打印执行之后,**p已经指向了0x7fff5fbff7c8的位置,此时的值是3,但是打印之前又进行了前++的操作,所以指针先向后移了一位,再进行打印。此时**p已经指向了0x7fff5fbff7d0,打印结果是5。
最终的结果是1 1 5。