• 【C语言进阶】指针进阶(三)


    9.指针和数组笔试题解析

    数组名的理解
    数组名是数组首元素的地址
    但是有2个例外:

    1. sizeof(数组名),这里的数组名表示整个数组,sizeof(数组名)计算的是整个数组的大小,单位是字节
    2. &数组名,这里的数组名表示整个数组,&数组名取出的是数组的地址
    #include
    int main()
    {
    	//一维数组
    	int a[] = { 1,2,3,4 };//4个元素,每个元素使int类型(4个字节)
    
    	printf("%d\n", sizeof(a));//16,数组名a单独放在sizeof内部,数组名表示整个数组,计算的是整个数组的大小单位是字节,是16字节
    	printf("%d\n", sizeof(a + 0));//a并非单独放在sizeof内部,也没有&,所以数组名a是数组首元素的地址,a+0还是首元素的地址
    	//是地址大小就是4/8 Byte
    
    	printf("%d\n", sizeof(*a));//a并非单独放在sizeof内部,也没有&,所以数组名a是数组首元素的地址
    	//*a 就是 首元素,大小就是4Byte  //*a == *(a+0) == a[0]
    
    	printf("%d\n", sizeof(a + 1));//a并非单独放在sizeof内部,也没有&,所以数组名a是数组首元素的地址,a+1就是第二个元素的地址
    	//a+1 == &a[1]  是第2个元素的地址,是地址就是4/8个字节
    
    	printf("%d\n", sizeof(a[1]));//a[1]就是数组的第二个元素,这里计算的就是第二个元素的大小,单位是字节 - 4
    
    	printf("%d\n", sizeof(&a));//&a - 是取出数组的地址,但是数组的地址也是地址,是地址就是4/8个Byte
    	//数组的地址 和 数组首元素的地址 的本质区别是类型的区别,并非大小的区别
    	//a  -- int*             int * p = a;
    	//&a -- int (*)[4]       int (*p)[4] = &a;
    	
    	printf("%d\n", sizeof(*&a));//16   对数组指针解引用访问一个数组的大小,单位是字节
    	//sizeof(*&a) --- sizeof(a) //16
    
    	printf("%d\n", sizeof(&a + 1));//&a数组的地址,&a+1还是地址,是地址就是4/8个字节
    
    	printf("%d\n", sizeof(&a[0]));//&a[0]是首元素的地址, 计算的是地址的大小 4/8 个字节
    	printf("%d\n", sizeof(&a[0] + 1));//&a[0]是首元素的地址,&a[0]+1就是第二个元素的地址,大小4/8个字节
    	//&a[1]
    	//&a[0]+1
    	//a+1
    
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36

    运行结果
    在这里插入图片描述

    #include
    #include 
    int main()
    {
    	//字符数组
    	char arr[] = { 'a','b','c','d','e','f' };//6
    
    	printf("%d\n", sizeof(arr));//6 数组名arr单独放在sizeof内部,计算的是整个数组的大小,单位是字节
    	printf("%d\n", sizeof(arr + 0));//arr是首元素的地址==&arr[0],是地址就是4/8个字节
    	//char* 
    	//指针变量的大小和类型无关,不管什么类型的指针变量,大小都是4/8个字节
    	//指针变量是用来存放地址的,地址存放需要多大空间,指针变量的大小就是几个字节
    	//32位环境下,地址是32个二进制位,需要4个字节,所以指针变量的大小就是4个字节
    	//64位环境下,地址是64个二进制位,需要8个字节,所以指针变量的大小就是8个字节
    	//门缝里看指针,把指针给看扁了
    	//
    	printf("%d\n", sizeof(*arr));//arr是首元素的地址,*arr就是首元素,大小就是1Byte
    	printf("%d\n", sizeof(arr[1]));//1
    	printf("%d\n", sizeof(&arr));//&arr是数组的地址,sizeof(&arr)就是4/8个字节
    	printf("%d\n", sizeof(&arr + 1));//&arr+1 是跳过数组后的地址,是地址就是4/8个字节
    	printf("%d\n", sizeof(&arr[0] + 1));//第二个元素的地址,是地址就是4/8Byte
    
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    运行结果:
    在这里插入图片描述

    //strlen 求字符串长度
    //统计的是在字符串中\0之前出现的字符的个数
    //
    int main()
    {
    	//字符数组
    	char arr[] = { 'a','b','c','d','e','f' };//6
    
    	printf("%d\n", strlen(arr));//随机值,arr是首元素的地址
    	printf("%d\n", strlen(arr + 0));//随机值,arr是首元素的地址, arr+0还是首元素的地址
    	//printf("%d\n", strlen(*arr));//err,arr是首元素的地址, *arr就是首元素 - 'a' - 97
    	//站在strlen的角度,认为传参进去的'a'-97就是地址,97作为地址,直接进行访问,就是非法访问
    	//printf("%d\n", strlen(arr[1]));//err, 'b' - 98
    	printf("%d\n", strlen(&arr));//随机值
    	//&arr -- char (*)[6]
    	//const char*
    	printf("%d\n", strlen(&arr + 1));//随机值
    	printf("%d\n", strlen(&arr[0] + 1));//随机值
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    运行结果:
    在这里插入图片描述

    int main()
    {
    	char arr[] = "abcdef";
    	//a b c d e f \0
    	printf("%d\n", sizeof(arr));//7数组名arr单独放在sizeof内部,计算的是整个数组的大小,单位是字节
    	printf("%d\n", sizeof(arr + 0));//arr是首元素的地址==&arr[0],是地址就是4/8个字节
    	printf("%d\n", sizeof(*arr));//arr是首元素的地址,*arr就是首元素,大小就是1Byte
    	printf("%d\n", sizeof(arr[1]));//1
    	printf("%d\n", sizeof(&arr));//4/8 &arr是数组的地址,sizeof(&arr)就是4/8个字节
    	printf("%d\n", sizeof(&arr + 1));//4/8 &arr+1 是跳过数组后的地址,是地址就是4/8个字节
    	printf("%d\n", sizeof(&arr[0] + 1));//4/8 第二个元素的地址,是地址就是4/8Byte
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    运行结果:
    在这里插入图片描述

    #include
    int main()
    {
    	char arr[] = "abcdef";
    	printf("%d\n", strlen(arr));//6 arr是首元素的地址
    	printf("%d\n", strlen(arr + 0));//6 arr是首元素的地址, arr+0还是首元素的地址
    	//printf("%d\n", strlen(*arr));//err arr是首元素的地址, *arr就是首元素 - 'a' - 97
    	//printf("%d\n", strlen(arr[1]));//err 'b' - 98
    	printf("%d\n", strlen(&arr));//6 
    	printf("%d\n", strlen(&arr + 1));//随机值
    	printf("%d\n", strlen(&arr[0] + 1));//5
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    运行结果:
    在这里插入图片描述

    int main()
    {
    	char* p = "abcdef";
    	printf("%d\n", sizeof(p));//4/8 计算的是指针变量的大小
    	printf("%d\n", sizeof(p + 1));//p+1还是地址,大小是4/8个字节
    	printf("%d\n", sizeof(*p));//1个字节, *p == 'a'
    	printf("%d\n", sizeof(p[0]));//1个字节, p[0]--> *(p+0) --> *p == 'a';
        printf("%d\n", sizeof(&p));//4/8个字节,&p 是地址
    	printf("%d\n", sizeof(&p + 1));//&p是地址,&p+1还是地址,是地址就是4/8个字节
    	printf("%d\n", sizeof(&p[0] + 1));//4/8
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    运行结果:
    在这里插入图片描述

    int main()
    {
    	char* p = "abcdef";
    	printf("%d\n", strlen(p));//6
    	printf("%d\n", strlen(p + 1));//5
    	//printf("%d\n", strlen(*p));//err
    	//printf("%d\n", strlen(p[0]));//err
    	printf("%d\n", strlen(&p));//随机
    	printf("%d\n", strlen(&p + 1));//随机
    	printf("%d\n", strlen(&p[0] + 1));//5
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    在这里插入图片描述

    运行结果:
    在这里插入图片描述

    int main()
    {
    	int a[3][4] = { 0 };
    
    	printf("%zd\n", sizeof(a));//48-数组名a单独放在了sizeof内存,表示整个数组,sizeof(a)计算的是数组的大小,单位是字节
    	printf("%zd\n", sizeof(a[0][0]));//4-a[0][0]是数组的第一行第一个元素,这里计算的就是一个元素的大小,单位是字节
    	printf("%zd\n", sizeof(a[0]));//16 - a[0]是第一行这个一维数组的数组名,数组名单独放在了sizeof内部
    	//a[0]就表示整个第一行这个一维数组,sizeof(a[0])计算的整个第一行这个一维数组的大小
    
    	printf("%zd\n", sizeof(a[0] + 1));//4/8 - a[0]并非单独放在sizeof内部,也没有&,所以a[0]表示第一行这个一维数组首元素的地址
    	//也就是第一行第一个元素的地址
    	//a[0] <---> &a[0][0]
    	//a[0]+1 ---> &a[0][1]
    	printf("%zd\n", sizeof(*(a[0] + 1)));//4 - a[0] + 1是第一行第二个元素的地址,*(a[0] + 1))就是第一行第二个元素
    	//
    	printf("%zd\n", sizeof(a + 1));//4/8
    	//a 作为二维数组的数组名,并没有单独放在sizeof内部,也没有&,a就是数组首元素的地址,也就是第一行的地址, a 的类型是 int(*)[4]
    	//a+1 就是第二行的地址,类型是:int(*)[4]
    	//
    	printf("%zd\n", sizeof(*(a + 1)));//16 a+1是第二行的地址,*(a+1)就是第二行,计算的就是第二行的大小
    	//另外一个角度理解:*(a+1) -- a[1]
    	//sizeof(a[1]) - a[1]这个第二行的数组名,单独放在了sizeof内部,计算的是第二行的大小
    
    	printf("%zd\n", sizeof(&a[0] + 1));//4/8
    	//a[0]是第一行的数组名,&a[0]取出的是数组的地址,取出的是第一行这个一维数组的地址,类型就是int(*)[4]
    	//&a[0]+1 就是第二行的地址,类型就是int(*)[4]
    	printf("%zd\n", sizeof(*(&a[0] + 1)));//*(&a[0] + 1)得到的就是第二行,计算的就是第二行的大小
    
    	printf("%zd\n", sizeof(*a));//16
    	//a表示数组首元素的地址,也就是第一行的地址
    	//*a 就是第一行,也就相当于是第一行的数组名
    	//*a--> *(a+0) -- a[0]
    	//
    	printf("%zd\n", sizeof(a[3]));//16-不会越界,
    	//a[3] --    arr[0]
    	//int [4]    int [4]
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38

    运行结果:
    在这里插入图片描述

    10.指针笔试题

    int main()
    {
    	int a[5] = { 1, 2, 3, 4, 5 };
    	int* ptr = (int*)(&a + 1);
    	printf("%d, %d", *(a + 1), *(ptr - 1));
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    运行结果:
    在这里插入图片描述

    //由于还没学习结构体,这里告知结构体的大小是20个字节
    //X86 环境下演示
    struct Test
    {
    	int Num;
    	char* pcName;
    	short sDate;
    	char cha[2];
    	short sBa[4];
    } * p;
    //假设p 的值为0x100000。 如下表表达式的值分别为多少?
    //已知,结构体Test类型的变量大小是20个字节
    //0x开头的数字是16进制的数字
    int main()
    {
    	p = (struct Test*)0x100000;
    	printf("%p\n", p + 0x1);
    	printf("%p\n", (unsigned long)p + 0x1);
    	printf("%p\n", (unsigned int*)p + 0x1);
    
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    运行结果:
    在这里插入图片描述

    int main()
    {
    	int a[4] = { 1, 2, 3, 4 };
    	int* ptr1 = (int*)(&a + 1);
    	int* ptr2 = (int*)((int)a + 1);
    	printf("%x,%x", ptr1[-1], *ptr2);
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    运行结果:
    在这里插入图片描述

    #include 
    int main()
    {
    	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
    	int* p;
    	p = a[0];
    	printf("%d", p[0]);//?
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    运行结果:
    在这里插入图片描述

    int main()
    {
    	int a[5][5];
    	int(*p)[4];//数组指针
    	p = a;//类型合适吗?   
    	//a - int(*)[5]
    	//p - int(*)[4]
    	printf("%p, %d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
    	//
    	//%p是打印地址,认为内存中存储的补码就是地址
    	//
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    运行结果:
    在这里插入图片描述

    int main()
    {
    	int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
    	int* ptr1 = (int*)(&aa + 1);
    	int* ptr2 = (int*)(*(aa + 1));
    	printf("%d,%d", *(ptr1 - 1), *(ptr2 - 1));
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    运行结果:
    在这里插入图片描述

    #include 
    //阿里的笔试题
    int main()
    {
    	char* a[] = { "work","at","alibaba" };
    	char** pa = a;
    	pa++;
    	printf("%s\n", *pa);
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    在这里插入图片描述

    int main()
    {
    	char* c[] = { "ENTER","NEW","POINT","FIRST" };
    	char** cp[] = { c + 3,c + 2,c + 1,c };
    	char*** cpp = cp;
    
    	printf("%s\n", **++cpp);
    	printf("%s\n", *-- * ++cpp + 3);
    	printf("%s\n", *cpp[-2] + 3);
    	printf("%s\n", cpp[-1][-1] + 1);
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    在这里插入图片描述

  • 相关阅读:
    基于spring boot开发的快递管理系统开题报告
    【计算机毕业设计】11.毕业生信息管理系统+vue
    深度学习-物体检测SSD
    淘宝 NPM 镜像站切换新域名啦
    普中51单片机:串口通信原理与应用指南(八)
    JAVA基础-11-Java Number 类--九五小庞
    将Sketch文件转化为PSD文件的简单在线工具!
    explain的执行结果:Impossible WHERE
    几种绘制时间线图的方法
    Pytorch_course1
  • 原文地址:https://blog.csdn.net/2201_75642960/article/details/133936447