指针和数组笔试题深度解析

学习之前我们将运用到sizeofstrlen函数

sizeofstrlen函数认识和区别:

1.strlen是一个库函数使用时需要引用#include<string.h>这个头文件,而sizeof是一个运算符号;

2.strlen计算的是'\0'之前的字符个数,sizefo计算的是所占空间内存的大小,单位是字节;

3.strlen计算时不包含'\0',而sizeof包含'\0';

4.strlen遇到'\0'才结束;

5.sizeof的类型是unsigned int ,是一个无符号的整型;

6.strlen只能用char做参数,sizeof可以用类型做参数;

注 数组名的意义:

  1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。

  2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。

  3. 除此之外所有的数组名都表示首元素的地址

接下来我们将进行几组练习,为了我们能进一步更好的认识

因为有32位平台和64位平台 所以占的内存是4/8

一维数组:

cpp 复制代码
//一维数组
	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]));
	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));

接下来我会一一分析他们

cpp 复制代码
int a[] = { 1,2,3,4 };//4个元素 每个元素为int (4字节)
cpp 复制代码
printf("%d\n", sizeof(a));//16

数组名a单独放在sizeof内部,数组名表示整个数组

计算的是整个数组的大小 所以是4*4=16个字节

cpp 复制代码
printf("%d\n", sizeof(a + 0));//4/8

数组名a没有单独放在sizeof内部,也没有&地址,所以数组名a是首元素的地址

计算a+0还是首元素的地址 所以就是4个字节

cpp 复制代码
	printf("%d\n", sizeof(*a));//4/8

a并非单独放在sizeof内部,也没有&a,所以数组名a是数组首元素的地址

*a就是对于第一个地址解引用 得到的第一个元素 大小就是4个字节

注:*a == *(a+0) == a0

cpp 复制代码
	printf("%d\n", sizeof(a + 1));//4/8

a并非单独放在sizeof内部,也没有&a,所以数组名a是数组首元素的地址

计算a+1就是第二个元素的地址 大小就是4个字节

cpp 复制代码
	printf("%d\n", sizeof(a[1]));//4/8

a1就是数组的第二个元素 大小就是4个字节

cpp 复制代码
	printf("%d\n", sizeof(&a));//4/8

&a - 是取出数组的地址,但是数组的地址也是地址,是地址就是4/8个字节

注:数组的地址 和 数组首元素的地址 的本质区别是类型的区别,并非大小的区别 地址就是4个字节

cpp 复制代码
	printf("%d\n", sizeof(*&a));//16

*和&同时存在会抵消,计算的是sizeof(a)的大小 所以还是16个字节

cpp 复制代码
	printf("%d\n", sizeof(&a + 1));//  4/8

&a数组的地址,&a+1还是地址,是地址就是4/8个字节

cpp 复制代码
	printf("%d\n", sizeof(&a[0]));//4/8

&a0是首元素的地址, 计算的是地址的大小 4/8 个字节

cpp 复制代码
	printf("%d\n", sizeof(&a[0] + 1));//4/8

/&a0是首元素的地址,&a0+1就是第二个元素的地址,大小4/8个字节


字符数组

字符数组第一种类型

cpp 复制代码
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));
cpp 复制代码
	char arr[] = { 'a','b','c','d','e','f' };// 6个元素 元素类型为char
cpp 复制代码
	printf("%d\n", sizeof(arr));//6

数组名arr单独放在sizeof内部,计算的是整个数组的大小 6*1

cpp 复制代码
	printf("%d\n", sizeof(arr + 0));//4/8

arr是首元素的地址==&arr0,是地址就是4/8个字节

cpp 复制代码
	printf("%d\n", sizeof(*arr));// 1

arr是首元素的地址,*arr就是首元素,大小就是1个字节

cpp 复制代码
	printf("%d\n", sizeof(arr[1]));//1

表示第二个元素 大小就是1个字节

cpp 复制代码
	printf("%d\n", sizeof(&arr));//4/8

&arr是数组的地址,地址就是4/8个字节

cpp 复制代码
	printf("%d\n", sizeof(&arr + 1));//4/8

&arr+1 是跳过数组后的地址,是地址就是4/8个字节

cpp 复制代码
	printf("%d\n", sizeof(&arr[0] + 1));//4/8

第二个元素的地址,是地址就是4/8个字节

cpp 复制代码
    char arr[] = { '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));
cpp 复制代码
	printf("%d\n", strlen(arr));// 随机值

随机值,arr是首元素的地址

注:strlen 求字符串长度 统计的是在字符串中\0之前出现的字符的个数

cpp 复制代码
随机值,arr是首元素的地址, arr+0还是首元素的地址//随机值

随机值,arr是首元素的地址, arr+0还是首元素的地址

cpp 复制代码
	printf("%d\n", strlen(*arr));// error

打印错误 arr是首元素的地址, *arr就是首元素 - 'a' - 97

站在strlen的角度,认为传参进去的'a'-97就是地址,97作为地址,直接进行访问,就是非法访问

cpp 复制代码
	printf("%d\n", strlen(arr[1]));//error

打印错误 'b' - 98 跟上题一样

cpp 复制代码
	printf("%d\n", strlen(&arr));//随机值

随机值,arr是首元素的地址

cpp 复制代码
	printf("%d\n", strlen(&arr + 1));//随机值-6

&arr+1跳过了整个数组 数组大小为6 所以是随机值-1

cpp 复制代码
	printf("%d\n", strlen(&arr[0] + 1));//随机值-1

&arr0+1为第二个元素的地址 所以是随机值减1


字符数组第二种类型

基本相似这俩种情况

1.char arr\[\]= {'a','b','c','d','e','f'};

a b c d e f

2.char arr\[\] = "abcdef";

a b c d e f \0

多了一个\0

第二种strlen(arr)就不会出现随机值

cpp 复制代码
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));
cpp 复制代码
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));

字符指针

cpp 复制代码
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));

这种也比较简单 要么是地址要么就是解引用得到的元素


cpp 复制代码
    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));

第一个和第二个和第七个都是表示首元素的地址 第三和第四都是解引用,打印错误

第五和第六都是随机值

&p 已经是一个新的变量了 不知道在哪里遇见\0 所以就是随机值


二维数组

cpp 复制代码
//二维数组
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]));

二维数组其实就是一维数组的连续存放

cpp 复制代码
	printf("%d\n", sizeof(a));//48

数组名a单独放在了sizeof内存,表示整个数组,sizeof(a)计算的是数组的大小,大小为48个字节

cpp 复制代码
	printf("%d\n", sizeof(a[0][0]));//4/8

a00是数组的第一行第一个元素,这里计算的就是一个元素的大小 大小为4个字节

cpp 复制代码
	printf("%d\n", sizeof(a[0]));//16

a0是第一行这个一维数组的数组名,数组名单独放在了sizeof内部

a0就表示整个第一行这个一维数组,sizeof(a0)计算的整个第一行这个一维数组的大小

所以大小为4*4=16个字节

cpp 复制代码
	printf("%d\n", sizeof(a[0] + 1));//4/8

a0并非单独放在sizeof内部,也没有&a,所以a0表示第一行这个一维数组首元素的地址

也就是第一行第一个元素的地址 &a00

a0 <---> &a00

a0+1 <---> &a01 大小也为4个字节

cpp 复制代码
	printf("%d\n", sizeof(*(a[0] + 1)));//4/8

a0 + 1是第一行第二个元素的地址,*(a0 + 1))就是第一行第二个元素

cpp 复制代码
	printf("%d\n", sizeof(a + 1));//4/8

a 作为二维数组的数组名,并没有单独放在sizeof内部,也没有&,a就是数组首元素的地址,也就是第一行的地址, a 的类型是 int(*)4

a+1 就是第二行的地址

地址就是4个字节

cpp 复制代码
	printf("%d\n", sizeof(*(a + 1)));//16

a+1是第二行的地址,*(a+1)就是第二行,计算的就是第二行的大小 16

cpp 复制代码
	printf("%d\n", sizeof(&a[0] + 1));//4/8

a0是第一行的数组名,&a0取出的是数组的地址,取出的是第一行这个一维数组的地址,类型就是int(*)4

&a0+1 就是第二行的地址,类型就是int(*)4

大小还是4个字节

cpp 复制代码
	printf("%d\n", sizeof(*(&a[0] + 1)));// 16

*(&a0 + 1)得到的就是第二行,计算的就是第二行的大小 16

cpp 复制代码
	printf("%d\n", sizeof(*a));//16

a表示数组首元素的地址,也就是第一行的地址

*a 就是第一行,也就相当于是第一行的数组名

*a--> *(a+0) -- a0 16

cpp 复制代码
	printf("%d\n", sizeof(a[3]));//16

不会越界 相当于a0


总结

其实今天的内容还是很好理解的不懂可以反复观看,看到这里的宝子们支持一下!

最后不要忘记一键三连阿!

我们下期再见,谢谢大家!

相关推荐
AIMath~4 分钟前
python中的uv命令揭秘
开发语言·python·uv
弹简特6 分钟前
【零基础学Python】06-Python模块和包、异常处理、文件常用操作
开发语言·python
x***r1518 分钟前
Postman-win64-7.2.2-Setup安装步骤详解(附API接口测试与参数配置教程)
开发语言·lua
吴可可12315 分钟前
CAD2004自定义实体开发环境配置
c++·算法
装不满的克莱因瓶16 分钟前
矩阵的主成分是什么?主成分分析(PCA)又能做什么?
人工智能·线性代数·算法·机器学习·ai·矩阵·pca
念恒1230617 分钟前
Python 面向对象编程核心:对象、实例化、封装与变量作用域
开发语言·python
大菜菜小个子21 分钟前
template<typename T>使用
java·开发语言·算法
L_090724 分钟前
【C++】C++11 新特性
开发语言·c++
方也_arkling27 分钟前
【Java-Day15】API篇-ArrayList集合
java·开发语言
我是一颗柠檬31 分钟前
【Java后端技术亮点】动态路由权限(按钮级权限),细粒度控制到按钮级别
java·开发语言·后端·状态模式