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

2023-09-19 14:39:41

学习之前我们将运用到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

一维数组:

//一维数组
	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));

接下来我会一一分析他们

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

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

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

printf("%d\n", sizeof(a + 0));//4/8

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

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

	printf("%d\n", sizeof(*a));//4/8

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

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

注:*a == *(a+0) == a[0]

	printf("%d\n", sizeof(a + 1));//4/8

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

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

	printf("%d\n", sizeof(a[1]));//4/8

 a[1]就是数组的第二个元素  大小就是4个字节

	printf("%d\n", sizeof(&a));//4/8

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

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

	printf("%d\n", sizeof(*&a));//16

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

	printf("%d\n", sizeof(&a + 1));//  4/8

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

	printf("%d\n", sizeof(&a[0]));//4/8

 &a[0]是首元素的地址, 计算的是地址的大小 4/8 个字节

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

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


字符数组

字符数组第一种类型

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

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

	printf("%d\n", sizeof(arr + 0));//4/8

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

	printf("%d\n", sizeof(*arr));// 1

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

	printf("%d\n", sizeof(arr[1]));//1

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

	printf("%d\n", sizeof(&arr));//4/8

 &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/8个字节 

    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));
	printf("%d\n", strlen(arr));// 随机值

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

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

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

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

	printf("%d\n", strlen(*arr));// error

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

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

	printf("%d\n", strlen(arr[1]));//error

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

	printf("%d\n", strlen(&arr));//随机值

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

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

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

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

 &arr[0]+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)就不会出现随机值

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));

 

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));


字符指针

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));



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


    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 所以就是随机值


二维数组

//二维数组
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]));

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

	printf("%d\n", sizeof(a));//48

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

	printf("%d\n", sizeof(a[0][0]));//4/8

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

	printf("%d\n", sizeof(a[0]));//16

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

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

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

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

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

也就是第一行第一个元素的地址 &a[0][0]

a[0] <---> &a[0][0]

a[0]+1 <---> &a[0][1]   大小也为4个字节

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

 a[0] + 1是第一行第二个元素的地址,*(a[0] + 1))就是第一行第二个元素

	printf("%d\n", sizeof(a + 1));//4/8

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

a+1 就是第二行的地址

地址就是4个字节

	printf("%d\n", sizeof(*(a + 1)));//16

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

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

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

&a[0]+1 就是第二行的地址,类型就是int(*)[4]

大小还是4个字节

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

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

	printf("%d\n", sizeof(*a));//16

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

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

*a--> *(a+0) -- a[0]    16

	printf("%d\n", sizeof(a[3]));//16

 不会越界 相当于a[0] 

 


总结

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

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

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

更多推荐

JavaScript面试题整理(二)

数据类型篇13、其他值到字符串的转换规则?Null和Undefined类型,null转换为‘null’,undefined转换为‘undefined’Boolean类型,true转换为‘true’,false转换为‘false’Number类型的值直接转换,不过那些极小和极大的数字会使用指数形式Symbol类型的值直接

React(react18)中组件通信04——redux入门

React(react18)中组件通信04——redux入门1.前言1.1React中组件通信的其他方式1.2介绍redux1.2.1参考官网1.2.2redux原理图1.2.3redux基础介绍1.2.3.1action1.2.3.2store1.2.3.3reducer1.3安装redux2.redux入门例子3.

算法、数据结构、计算机系统、数据库MYSQL、概率论、数学实验MATLAB、数学建模、马原、英语、杂项、QT项目

算法冒号表达式(condition)?x:y可以三个条件以此类推(condition1)?x:(condition2)?y:z判断三角形最简单的办法boolcanFormTriangle(inta,intb,intc){return(a+b>c)&&(b+c>a)&&(a+c>b);}带空格的数据输入#include<

【ROS入门】创建工作空间与功能包

文章结构工作空间文件结构创建工作空间流程创建工作空间编译工作空间设置环境变量/创建功能包创建功能包编译功能包检查环境变量工作空间文件结构工作空间(workspace)是一个存放工程开发相关文件的文件夹,类似于在windows中使用IDE创建的工程。主要分为以下四个文件夹:src:代码空间(SourceSpace):用于

Java版分布式微服务云开发架构 Spring Cloud+Spring Boot+Mybatis 电子招标采购系统功能清单

项目说明随着公司的快速发展,企业人员和经营规模不断壮大,公司对内部招采管理的提升提出了更高的要求。在企业里建立一个公平、公开、公正的采购环境,最大限度控制采购成本至关重要。符合国家电子招投标法律法规及相关规范,以及审计监督要求;通过电子化平台提高招投标工作的公开性和透明性;通过电子化招投标,使得招标采购的质量更高、速度

【2023,学点儿新Java-27】是的——C语言中的const关键字 | 附:按照类型 快速了解与划分:C语言中的关键字 | goto关键字解释

前情回顾:【2023,学点儿新Java-26】关键字介绍+示例代码:assert断言(如何启用断言),以验证一个数组的长度是否不为零为例说明【2023,学点儿新Java-25】如何解决浮点计算存在误差:strictfp|如何保护敏感信息,提高程序的可靠性和安全性:transient|附:Java异常处理关键字介绍【20

Spring Boot启动源码分析

一,前言版本:spring-boot-starter-parent版本为2.3.0SpringBoot项目的启动入口是一个main方法,因此我们从该方法入手即可二,源码分析跟踪run方法/***SpringApplication的方法*@paramprimarySource启动类的class*@paramargs启动参

GPT,GPT-2,GPT-3,InstructGPT的进化之路

ChatGPT火遍圈内外,突然之间,好多人开始想要了解NLP这个领域,想知道ChatGPT到底是个什么?作为在这个行业奋斗5年的从业者,真的很开心让人们知道有一群人在干着这么样的一件事情。这也是我结合各位大佬的文章,总结下GPT这条技术路线的初心。其实,ChatGPT的成功并非一朝一夕,而是OpenAI长达4年多持续努

基于Uniapp+SpringBoot+Vue的电影交流平台小程序设计与实现(源码+lw+部署文档+讲解等)

前言💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌💗👇🏻精彩专栏推荐订阅👇🏻2023-2024年最值得选的微信小程序毕业设计选题大全:100个热门选

GaussDB数据库SQL系列-自定义函数

目录一、前言二、自定义函数(Function)概述三、使用场景四、优缺点1、数据库中Function的使用优点2、数据库中Function的使用缺点五、GaussDB中的Function示例与解析1、示例一:定义函数为SQL查询2、示例二:返回一个包含多个输出参数的记录3、示例三:返回RECORD类型结果集六、小结一、

【数据结构】平衡二叉搜索树(AVL树)——AVL树的概念和介绍、AVL树的简单实现、AVL树的增删查改

文章目录平衡二叉搜索树(AVL树)1.AVL树的概念和介绍2.AVL树的简单实现2.1AVL树的插入2.2AVL树的旋转2.2.1左旋2.2.2右旋2.2.3右左双旋2.2.4左右双旋全部源码平衡二叉搜索树(AVL树)为什么要引入平衡二叉搜索树?在之前我们学习了二叉搜索树,二叉搜索树的结构类似于一个倒置的树,而左子树的

热文推荐