sizeof与strlen练习

news2025/1/16 11:14:49

前言

在这里插入图片描述
本篇仅仅是为了更加了解sizeof操作符和strlen函数练习.
对于多条sizeof操作符和strlen函数出现,可能很容易造成头脑不清晰,做题时容易混乱.

目录

  • 前言
  • 一维数组
  • 字符数组
    • 情况1:
    • 情况2
    • 情况3
  • 二维数组

练习之前请牢记下面这段话.这将是头脑清晰地关键.
提示:

  1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
  2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
  3. 除此之外所有的数组名都表示首元素的地址。

sizeof只关心()里面的类型,并不关心里面放的是什么,这点希望做题目时可以牢记.
strlen()括号里面的是地址,它只会通过访问括号里面的地址,往后一直寻找’\0’,直到’\0’才会停止.

一维数组

#include <stdio.h>
int main()
{
	int a[] = { 1,2,3,4 };
	printf("%d\n", sizeof(a));			//1
	printf("%d\n", sizeof(a + 0));		//2
	printf("%d\n", sizeof(*a));			//3
	printf("%d\n", sizeof(a + 1));		//4
	printf("%d\n", sizeof(a[1]));		//5
	printf("%d\n", sizeof(&a));			//6
	printf("%d\n", sizeof(*&a));		//7
	printf("%d\n", sizeof(&a + 1));		//8
	printf("%d\n", sizeof(&a[0]));		//9
	printf("%d\n", sizeof(&a[0] + 1));	//10
	return 0;
}

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

1	->	16
2	->	8
3	->	4
4	->	8
5	->	4
6	->	8
7	->	16
8	->	8
9	->	8
10	->	8

详细分析答案:

1.	16字节:		数组名a单独出现在sizeof()的括号中表示计算整个数组的大小,4*4=16字节
2.	4/8字节: 	a数组名不是单独出现,+0表示首地址表示表示首元素地址,地址的大小4/8字节
3.	4字节:		(*a)解引用得到的是数组首元素数组首元素是1,类型为整形,整形大小是4字节.
4.	4/8字节:	a+1表示数组第二个元素的地址.地址大小4/8字节.
5.	4字节:		a[1]表示数组的第二个元素,元素2的类型是整形,整形的大小是4字节.
6.	4/8字节:	&a表示是整个数组的地址,类型为int(*)[4].但是数组的地址也是地址,地址就是4/8字节.
7.	16字节		*&a表示对&a解引用,得到的是整个数组的地址.也可以理解为*&符号是可以抵消的,即等价于sizeof(a);结果为4*4=16字节
8.	4/8个字节:	&a+1表示指向数组最后一个元素的后面.因为&a的类型是int(*)[4],+1会跳过四个整形.
地址的大小就是4/8个字节.
9.	4/8字节:	&a[0]表示首元素的地址,地址大小为4/8字节.
10. 4/8字节:	(&a[0] + 1表示第二个元素的地址.4/8字节.

在这里插入图片描述

字符数组

情况1:

字符数组:

char arr[] = { ‘a’,‘b’,‘c’,‘d’,‘e’,‘f’ };

#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = { 'a','b','c','d','e','f' };
	printf("%d\n", sizeof(arr));			//1
	printf("%d\n", sizeof(arr + 0));		//2
	printf("%d\n", sizeof(*arr));			//3
	printf("%d\n", sizeof(arr[1]));			//4
	printf("%d\n", sizeof(&arr));			//5
	printf("%d\n", sizeof(&arr + 1));		//6
	printf("%d\n", sizeof(&arr[0] + 1));	//7
	printf("%d\n", strlen(arr));			//8
	printf("%d\n", strlen(arr + 0));		//9
	printf("%d\n", strlen(*arr));			//10
	printf("%d\n", strlen(arr[1]));			//11
	printf("%d\n", strlen(&arr));			//12
	printf("%d\n", strlen(&arr + 1));		//13
	printf("%d\n", strlen(&arr[0] + 1));	//14
	return 0;
}

在这里插入图片描述

答案分析:

1.	6字节:	 	表示整个数组的地址,1*6=6个字节
2. 	4/8字节	:	首元素地址是4/8个字节.
3. 	1字节:		*arr表示首元素是一个字符型,1个字节.
4.	1字节:		arr[1]表示数组的第二个元素,大小为1个字节.
5. 	4/8个字节:	&arr表示整个数组的地址,4/8个字节.
6.	4/8个字节:	&arr+1表示数组最后一个元素的后一个位置的地址.4/8个字节
7. 	4/8个字节:	表示第二个元素的地址.4/8个字节

strlen函数从参数处开始,遇到’\0’才会停止.

8.随机值.: 	由于该数组没有’\0,所以会向后一直找直到碰到'\0',
9.随机值:	arr+0同样表示从数组的第一个元素,从第一个元素往后找,与上面一个道理.
10.报错:	strlen()括号里面是地址,所以会访问地址名为a的地址,a的ASCII码值是’97,访问地址’97,系统会报错
11.报错:	同理,访问数组第二个元素,地址名b的地址,系统报错
12.随机值:	&arr取出的是整个数组的地址,地址还是从数组首地址开始往后找'\0'.
13.随机值:	&arr的类型是char(*)[6],+1会跳过这个数组从数组后面开始找'\0'.
14.随机值:	从数组的第二个元素开始找'\0'.

情况2

字符数组:

char arr[] = “abcdef”;

#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = "abcdef";
	printf("%d\n", sizeof(arr));			//1
	printf("%d\n", sizeof(arr + 0));		//2
	printf("%d\n", sizeof(*arr));			//3
	printf("%d\n", sizeof(arr[1]));			//4
	printf("%d\n", sizeof(&arr));			//5
	printf("%d\n", sizeof(&arr + 1));		//6
	printf("%d\n", sizeof(&arr[0] + 1));	//7
	printf("%d\n", strlen(arr));			//8
	printf("%d\n", strlen(arr + 0));		//9
	printf("%d\n", strlen(*arr));			//10
	printf("%d\n", strlen(arr[1]));			//11
	printf("%d\n", strlen(&arr));			//12
	printf("%d\n", strlen(&arr + 1));		//13
	printf("%d\n", strlen(&arr[0] + 1));	//14
	return 0;
}

答案分析:

1.	7字节:	 	表示整个数组的地址,1*7=7个字节
2. 	4/8字节:	首元素地址是4/8个字节.
3. 	1字节:		*arr表示首元素是,一个字符型,1个字节.
4.	1字节:		arr[1]表示数组的第二个元素,大小为1个字节.
5. 	4/8个字节:	&arr表示整个数组的地址,4/8个字节.
6.	4/8个字节:	&arr+1表示数组最后一个元素的后一个位置的地址4/8个字节
7. 	4/8个字节:	表示第二个元素的地址4/8个字节
8.	6字节: 		从数组首元素开始,直到最后一个元素’\0,6个元素.6*1=6字节.
9.	6字节:		arr+0同样表示从数组的第一个元素,从第一个元素往后找,与上面一个道理.
10.	报错:		strlen()括号里面是地址,所以会访问地址名为a的地址,a的ASCII码值是’97,访问地址’97,系统会报错
11.	报错:		同理,访问数组第二个元素,地址名b的地址,系统报错
12.	6字节:		&arr取出的是整个数组的地址,地址还是从数组首地址开始往后找'\0.
13.	随机值:		&arr的类型是char(*)[6],+1会跳过这个数组,从数组后面开始找'\0.
14.	5字节:		从数组的第二个元素'b'开始往后找'\0'.

情况3

char* p = “abcdef”;

#include <stdio.h>
#include <string.h>
int main()
{
	char* p = "abcdef";
	printf("%d\n", sizeof(p));				//1
	printf("%d\n", sizeof(p + 1));			//2
	printf("%d\n", sizeof(*p));				//3
	printf("%d\n", sizeof(p[0]));			//4
	printf("%d\n", sizeof(&p));				//5
	printf("%d\n", sizeof(&p + 1));			//6
	printf("%d\n", sizeof(&p[0] + 1));		//7
	printf("%d\n", strlen(p));				//8
	printf("%d\n", strlen(p + 1));			//9
	printf("%d\n", strlen(*p));				//10
	printf("%d\n", strlen(p[0]));			//11
	printf("%d\n", strlen(&p));				//12
	printf("%d\n", strlen(&p + 1));			//13
	printf("%d\n", strlen(&p[0] + 1));		//14
	return 0;
}

答案分析:
在这里插入图片描述

1.	4/8字节:	 p是一个指针变量,这里指向的是常量字符串,首元素’a’,指针大小4/8字节.
2.	4/8字节:	p+1表示指向第二个元素’b’,地址(指针)的大小为4/8字节.
3.	1字节:		*p表示首元素’a’,字符a是char类型,占一个字节.
4.	1字节:		p[0]同样是表示数组首元素,占一个字节.
5.	4/8个字节:	&p表示一级指针变量p的地址,类型为char**的二级指针,指针大小为4/8字节.
6.	4/8个字节:	二级指针&p+1,表示该二级指针的后一个地址,请看图.指针大小4/8字节.
7.	4/8个字节:	表示首元素地址+1,即第二个元素地址,4/8个字节.
8	6字节 		从字符串首元素开始,直到最后一个元素’\0,6个元素.6*1=6字节.
9	5字节		从字符串第二个元素往后,找’\0,’bcdef’共5个元素
10.	报错:		表示从a地址,97’地址往后找’\0,系统报错.
11.	报错:		同理, 也是访问地址名为’97’的地址.系统报错.
12.	随机值	 	&p表示一级指针变量p的地址,从存放指针’p’的地址往后找’\0.
13.	随机值:		&p+1表示从存放指针变量’p’的地址后面一个地址,找’\0.
14	5字节:		从字符串的第二个元素开始找'\0'.

二维数组

//二维数组
#include <stdio.h>
int main()
{
	int a[3][4] = { 0 };
	printf("%d\n", sizeof(a));						//1
	printf("%d\n", sizeof(a[0][0]));				//2
	printf("%d\n", sizeof(a[0]));					//3
	printf("%d\n", sizeof(a[0] + 1));				//4
	printf("%d\n", sizeof(*(a[0] + 1)));			//5
	printf("%d\n", sizeof(a + 1));					//6
	printf("%d\n", sizeof(*(a + 1)));				//7
	printf("%d\n", sizeof(&a[0] + 1));				//8
	printf("%d\n", sizeof(*(&a[0] + 1)));			//9
	printf("%d\n", sizeof(*a));						//10
	printf("%d\n", sizeof(a[3]));					//11
	return 0;
}

答案:
在这里插入图片描述

1.	48字节		a单独出现在sizeof括号中,表示计算整个数组的地址,数组每个元素是整形,4*3*4=48字节
2.	4字节		a[0][0]计算数组首元素的地址,是一个整形占四个字节.
3.	16字节		a[0]是第一行的数组名,数组名单独放在sizeof,表示第一行的地址,4*4=16字节
4.	4/8字节		a[0]不是单独出现,也没有&(取地址),表示的是第一行第一个元素即a[0][0],后面+1表示第一行第二个元素的地址.
5.	4字节		因为a[0]+1表示第一行第二个元素的地址,则解引用得到的是第一行第二个元素.该元素是整形.
6.	4/8字节		a是数组名,表示二维数组的第一行,a的类型是int(*)[4],那么+1会跳过一行,即表示第二行的地址,地址4/.8字节.
7.	16字节		*(a+1)等价于a[0],第二行的数组名单独放在sizeof里面,表示得到第二行所有元素.
8.	4/8字节		 (&a[0] + 1)),&a[0]表示第一行的地址,+1表示第二行的地址,地址是4/8字节
9.	16字节	 	对第二行解引用,等价于a[1],数组名单独放进sizeof,表示这一行的大小.
10.	16字节		a是表示第一行的地址,*a等价于a[0],类型是int(*)[4].
11. 16字节		a[3]虽然表示第四行数组名,二维数组只有3,但是sizeof并没有访问第四行,而是计算第四行类型大小,int(*)[4].

希望通过这些练习能更加清楚的指针,通过sizeof和strlen函数里面的参数,指针的来回变化会加深对指针的理解.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/387479.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MyBatis高频面试专题

一、介绍下MyBatis中的工作原理 1。介绍MyBatis的基本情况&#xff1a;ORM 2。原理&#xff1a; MyBatis框架的初始化操作处理SQL请求的流程 1.系统启动的时候会加载解析全局配置文件和对应映射文件。加载解析的相关信息存储在 Configuration 对象 Testpublic void test1(…

【ID:17】【20分】A. DS顺序表--类实现

时间限制1秒内存限制128兆字节题目描述用C语言和类实现顺序表属性包括&#xff1a;数组、实际长度、最大长度&#xff08;设定为1000&#xff09;操作包括&#xff1a;创建、插入、删除、查找类定义参考输入第1行先输入n表示有n个数据&#xff0c;即n是实际长度;接着输入n个数据…

HCIE-Cloud Computing LAB备考第二步:逐题攻破--第三题:迁移

迁移 题目 将一台AD服务器迁移到FusionCompute平台,并保障业务正常。 思维导图 markmap内容1 文字介绍 准备Rainbow服务器:在Windows系统安装Rainbow,必须保证其与源端主机、目的端平台互通。关闭防火墙。【首次登录rainbow时,需要注册用户名和密码,考试时根据考题要…

989. 数组形式的整数加法

989. 数组形式的整数加法https://leetcode.cn/problems/add-to-array-form-of-integer/ 难度简单226 整数的 数组形式 num 是按照从左到右的顺序表示其数字的数组。 例如&#xff0c;对于 num 1321 &#xff0c;数组形式是 [1,3,2,1] 。 给定 num &#xff0c;整数的 数组…

【Linux】进程等待 | 详解 wait/waitpid 的 status 参数

&#x1f923; 爆笑教程 &#x1f449; 《看表情包学Linux》&#x1f448; 猛戳订阅 &#x1f525; &#x1f4ad; 写在前面&#xff1a;在上一章中我们讲解了进程创建与进程终止&#xff0c;本章我们开始讲解进程等待。进程等待这部分知识相较于前面还是较为复杂的&#xff0…

Vue2的生命周期(详解)

Vue的生命周期一、生命周期的概念二、钩子函数三、Vue2的生命周期3.1 初始化阶段3.2 挂载阶段3.3 更新阶段3.4 销毁阶段一、生命周期的概念 Vue实例的生命周期: 从创建到销毁的整个过程 二、钩子函数 Vue框架内置函数,随着组件的生命周期阶段,自动执行 作用:特定的时间点,执行特…

Unity Lighting -- 向场景中添加光源

本节笔记来实践向场景中添加光源。 除了平行光源外&#xff0c;还有两种常用的光源&#xff1a; 点光源&#xff08;Point Lights&#xff09;&#xff1a;点光源所发出的光是朝四面八方发散的&#xff0c;我们可以用点光源来模拟灯泡之类的发光物体。 聚光灯源&#xff08;Spo…

python每日学9 : windows上配置gitee的远程仓库,git的初步使用

在开发中&#xff0c;如果遇到复杂的项目&#xff0c;使用版本控制是非常有必要的&#xff0c;如果涉及到多端开发&#xff0c;那么还需要使用远程仓库。本文作个简单记录&#xff0c;记录下git初步使用。 1 下载与安装 git还有几个ui版本&#xff0c;但是开始使用的话&#…

【LeetCode】带环链表两道题

第一题&#xff1a;环形链表 问题介绍 给你一个链表的头节点head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪next指针再次到达&#xff0c;则链表中存在环。为了表示给定链表中的环&#xff0c;评测系统内部使用整数pos 来表示链表…

【Storm】【三】Storm 核心概念详解

Storm 核心概念详解 一、Storm核心概念1.1 Topologies&#xff08;拓扑&#xff09;1.2 Streams&#xff08;流&#xff09;1.3 Spouts1.4 Bolts1.5 Stream groupings&#xff08;分组策略&#xff09;二、Storm架构详解2.1 nimbus进程2.2 supervisor进程2.3 zookeeper的作用2.…

【蒸滴C】C语言结构体入门?看这一篇就够了

目录 一、结构体的定义 二、结构的声明 例子 三、 结构成员的类型 结构体变量的定义和初始化 1.声明类型的同时定义变量p1 2.直接定义结构体变量p2 3.初始化&#xff1a;定义变量的同时赋初值。 4.结构体变量的定义放在结构体的声明之后 5.结构体嵌套初始化 6.结构体…

24节气-惊蛰 // 诗句、海报分享,春风至,惊雷动。

惊蛰&#xff0c;古称"启蛰"&#xff0c;是二十四节气中的第3个节气&#xff0c;更是干支历卯月的起始;时间点在公历3月5-6日之间&#xff0c;太阳到达黄经345时。《月令七十二候集解》:"二月节……万物出乎震&#xff0c;震为雷&#xff0c;故曰惊蛰&#xff0…

【ONE·C || 动态内存管理】

总言 C语言&#xff1a;动态内存管理介绍。 文章目录总言1、为什么存在动态内存管理2、动态内存函数介绍2.1、malloc、free2.1.1、malloc函数2.1.2、free函数2.2、calloc、realloc2.2.1、calloc函数2.2.2、realloc函数3、常见的动态内存错误3.1、对NULL指针的解引用操作3.2、对…

TEX:显示文本

文章目录字体选择字体fontspec宏包根据字体形状控制字体为不同的字体形状选择不同的特征为不同的字体大小状选择不同的特征中文字体选择xeCJK宏包字体选择与设置XELATEX字体名查找字体集与符号居中与缩进居中单边调整两边缩进诗歌缩进列表itemize样例enumerate样例description样…

Java多线程(三)——线程池及定时器

线程池就是一个可以复用线程的技术。前面三种多线程方法就是在用户发起一个线程请求就创建一个新线程来处理&#xff0c;下次新任务来了又要创建新线程&#xff0c;而创建新线程的开销是很大的&#xff0c;这样会严重影响系统的性能。线程池就相当于预先创建好几个线程&#xf…

concrt140.dll丢失四种方法解决丨提示游戏里找不到concrt140.dll?

电脑提示concrt140.dll文件丢失怎么办&#xff1f;由于找不到concrt140.dll&#xff0c;无法继续执行代码&#xff1f; 我们平时在打开 Adobe 应用程序、Halo、Forza Horizon 5 地平线5 等时&#xff0c;可能会遇到找不到 concrt140.dll。因此&#xff0c;这不是特定于某个应用…

基频的后处理

基频归一化 基频为什么要归一化&#xff1f;为了消除人际随机差异&#xff0c;提取恒定参数&#xff0c;在语际变异中找到共性。 引言 声调的主要载体就是基频。但是对声调的感知会因人而异&#xff0c;例如某个听感上的高升调&#xff0c;不同的调查人员可能会分别描写成 […

Nginx的负载均衡

Nginx不仅可以作为一个web服务器或反向代理服务器&#xff0c;还可以按照权重、轮询、ip_hash、URL_hash等多种方式实现对后端服务器的负载均衡。 负载均衡的概念&#xff1a; 负载均衡就是将负载分摊到多个操作单元上执行&#xff0c;从而提高服务的可用性和相应速度&#xf…

数据仓库、数据中台、数据湖都是什么?

相信很多人都在最近的招聘市场上看到过招聘要求里提到了数据仓库、数据中台&#xff0c;甚至还有数据湖&#xff0c;这些层出不穷的概念让人困扰。今天我就来跟大家讲一讲数据仓库、数据中台以及数据湖的概念及区别。 数据库 在了解数据仓库、数据中台以及数据湖之前&#xff…

JDBC

JDBC核心技术 讲师&#xff1a;宋红康 微博&#xff1a;尚硅谷-宋红康 第1章&#xff1a;JDBC概述 1.1 数据的持久化 持久化(persistence)&#xff1a;把数据保存到可掉电式存储设备中以供之后使用。大多数情况下&#xff0c;特别是企业级应用&#xff0c;数据持久化意味着将内…