本文章写于2023.2.9,一些指针练习题(本文章所有代码均在64位编译器下运行)
首先有一些重要的知识点:
假设a是一个数组
&a取出的是数组的地址
sizeof(a) 数组名表示整个数组-计算的是整个数组的大小,单位是字节
除此之外,其它情况的数组名a均为首元素地址
1
int a[] = {1,2,3,4};
printf("%d\n",sizeof(a));
printf("%d\n",sizeof(a+0));
printf("%d\n",sizeof(*a));
printf("%d\n",sizeof(a+1));
printf("%d\n",sizeof(a[1]));
1.16 sizeof(a)表示的是整个数组地址
2.8 sizeof(a+0)表示的是首元素的地址
3.4 a为首元素的地址,解引用为1,1是int类型,为4个字节
4.8 a+1表示第二个元素的地址
5.4 a[1]表示第2个元素,即2,是int类型,为4个字节
2
int a[] = {1,2,3,4};
printf("%d\n",sizeof(&a));
printf("%d\n",sizeof(*&a));
printf("%d\n",sizeof(&a + 1));
printf("%d\n",sizeof(&a[0]));
printf("%d\n",sizeof(&a[0] + 1));
8 &a取出的是整个数组的地址
16 &a取出的是整个数组地址,解引用得到整个数组,即sizeof(a) (也可以理解为*和&抵消了)
8 &a+1直接跳过这个数组,结果是个地址
8 取出的是1的地址
8 取出的是2的地址
3
char arr[] = {'a', 'b', 'c', 'd', 'e', 'f'};
printf("%d\n",sizeof(arr));
printf("%d\n",sizeof(arr+0));
printf("%d\n",sizeof(*arr));
printf("%d\n",sizeof(arr[1]));
printf("%d\n",sizeof(&arr));
printf("%d\n",sizeof(&arr + 1));
printf("%d\n",sizeof(&arr[0] + 1));
注意:arr数组里并没有'\0'
6
8 sizeof(arr+0)计算的是首元素的地址
1 *arr即为a,a是char类型
1 arr[1]即为b
8 &arr取出的是整个数组的地址,但仍然是个地址,结果为8
8 跳过这个arr数组
8 &arr[0]得到的是首元素的地址,+1指向第二个元素,即为'b'的地址
![](https://i-blog.csdnimg.cn/blog_migrate/2bb687c1d13b8063419ab01375ba65ff.png)
4
char arr[0] = {'a', 'b', 'c', 'd', 'e', 'f'};
printf("%d\n", strlen(arr));
printf("%d\n", strlen(arr + 0));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
printf("%d\n", strlen(&arr));
printf("%d\n", strlen(&arr + 1));
printf("%d\n", strlen(&arr[0] + 1));
注意:strlen遇'\n'才停止计数,且参数传递的是地址
随机值(此后简称s) 因为不知道什么时候会遇到'\n'
s
erro:因为*arr即为'a',非法访问地址
erro
s
s - 6
s - 1
5
char arr[] = "abcdef";
printf("%d\n",sizeof(arr));
printf("%d\n",sizeof(arr+0));
printf("%d\n",sizeof(*arr));
printf("%d\n",sizeof(arr[1]));
printf("%d\n",sizeof(&arr));
printf("%d\n",sizeof(&arr + 1));
printf("%d\n",sizeof(&arr[0] + 1));
7:sizeof(arr)表示的是整个数组的大小, 别忘了有'\0'
8:arr+0表示首元素的地址+0
1:*arr即为'a'
1
8
8
8
6
char arr[] = "abcdef";
printf("%d\n",strlen(arr));
printf("%d\n",strlen(arr+0));
printf("%d\n",strlen(*arr));
printf("%d\n",strlen(arr[1]));
printf("%d\n",strlen(&arr));
printf("%d\n",strlen(&arr + 1));
printf("%d\n",strlen(&arr[0] + 1));
6:不解释,不要算成7啦!!
6
erro
erro
6
s
5
7
char* p = "abcdef";
printf("%d\n", sizeof(p));
printf("%d\n", sizeof(p + 1));
printf("%d\n", sizeof(*p));
printf("%d\n", sizeof(p[0]));
printf("%d\n", sizeof(&p));
printf("%d\n", sizeof(&p+1));
printf("%d\n", sizeof(&p[0] + 1));
p是一个指针,储存的是a的地址
8:地址的大小是8
8:此时p指向b
1
1
8:&p为一个地址
8:&p+1为一个地址
8:&p[0]+1指向'b'
![](https://i-blog.csdnimg.cn/blog_migrate/41940746ca93787af07628e1128451b7.png)
8
char* p = "abcdef";
printf("%d\n", strlen(p));
printf("%d\n", strlen(p + 1));
printf("%d\n", strlen(*p));
printf("%d\n", strlen(p[0]));
printf("%d\n", strlen(&p));
printf("%d\n", strlen(&p+1));
printf("%d\n", strlen(&p[0] + 1));
6
5:此时p指向b
erro
erro
s
s
5
9.二维数组指针练习
int a[3][4] = {0};
printf("%d\n",sizeof(a));
printf("%d\n",sizeof(a[0][0]));
printf("%d\n",sizeof(a[0]));
printf("%d\n",sizeof(a[0]+1));
printf("%d\n",sizeof(*(a[0] + 1)));
printf("%d\n",sizeof(a+1));
printf("%d\n",sizeof(*(a+1)));
printf("%d\n",sizeof(&a[0]+1));
printf("%d\n",sizeof(*(&a[0]+1)));
printf("%d\n",sizeof(*a));
printf("%d\n",sizeof(a[3]));
48:3*4*12=48
4
16:sizeof(a[0])表示第一行大小
8:这时的a[0]表示第一行的首元素地址,+1指向第二个元素
4
8:此时的a为二维数组的首元素的地址,即第一行的地址,+1指向下一行
16:计算的是第二行的大小 或者这样理解:*(a+1) --> a[1]
8:a[0]是第一行的数组名,&a[0]取出的是第一行的地址,+1指向第二行
16:&a[0]+1是第二行的地址,计算的是第二行的大小
16:*a表示第一行,第一行的大小是16 或者这样理解:*(a+0) --> a[0]
16:假设有第四行,则该行的大小是16(sizeof括号里的东西不参与运算)
![](https://i-blog.csdnimg.cn/blog_migrate/b0a001281d2bc89f1503e207fc3876bb.png)
指针笔试题
指针笔试题1
int main()
{
int a[5] = {1,2,3,4,5};
int* ptr = (int *)(&a + 1);
printf("%d, %d",*(a + 1), *(ptr - 1));
return 0;
}
![](https://i-blog.csdnimg.cn/blog_migrate/5bee0063648ecfc553144593d5fd9243.png)
指针笔试题2
// 告知结构体大小为20个字节
struct Test
{
int Num;
char* pcName;
short sDate;
char cha[2];
short sBa[4];
}* p; //定义了一个结构体指针
// 假设p 的值为0x100000。如下表达式的值分别为多少?
int main()
{
printf("%p\n", p + 0x1);
printf("%p\n", (unsigned long)p + 0x1);
printf("%p\n", (unsigned int*)p + 0x1);
}
![](https://i-blog.csdnimg.cn/blog_migrate/f680e1b3152fc4b9580aa01eccb4f4b8.png)
指针笔试题3
int main()
{
int a[4] = { 1,2,3,4 };
int* ptr1 = (int*)(&a + 1);
int* ptr2 = (int*)((int)a + 1);
return 0;
}
![](https://i-blog.csdnimg.cn/blog_migrate/456156aad38264dfdbc109ee0b46e5b8.png)
笔试题4
int main()
{
int a[3][2] = { (0,1),(2,3), (4,5) };
int* p = a[0];
printf("%d", p[0]);
return 0;
}
![](https://i-blog.csdnimg.cn/blog_migrate/d84c9b95c973252446ca603d65ab7047.png)
笔试题5
int main()
{
int a[5][5];
int(*p)[4];
p = a;
printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
return 0;
}
![](https://i-blog.csdnimg.cn/blog_migrate/10c3041a1367df61a5c8f3219b735848.png)
笔试题6
#include <stdio.h>
int main()
{
int aa[2][5] = { 1,2,3,4,5,6,7,8,9,10 };
int* ptr1 = (int*)(&aa + 1);
int* ptr2 = (*(aa + 1));
printf("%d,%d\n", *(ptr1 - 1), *(ptr2 - 1));
return 0;
}
![](https://i-blog.csdnimg.cn/blog_migrate/b2d922714a91e8236eb068c0cf799a04.png)