首页 > 其他分享 >使用sizeof()和strlen()去计算【数组】和【指针】的大小

使用sizeof()和strlen()去计算【数组】和【指针】的大小

时间:2023-11-10 14:06:42浏览次数:37  
标签:arr 地址 数组 printf sizeof strlen 指针

(目录)

一、知识回顾

1、回顾sizeof()、strlen的作用:

  • sizeof()是用来求取 变量 或者 类型 所占内存空间的大小(单位:字节)。
  • sizeof计算的是占用内存空间的大小,单位是字节,不关注内存中到底存放的是什么。sizeof不是函数,是操作符
  • strlen()是一个库函数是专门用来计算 字符串 长度的,在对其进行调用前是需要包含头文件<string.h>。
  • strlen()函数是通过字符串结束标志 \0 来计算字符串长度的,但计算出来的字符串长度是不包括 \0 的,也就是说所谓的长度就是 \0 前字符的个数。strlen是函数

sizeof是一个运算符,用于计算一个变量或数据类型所占的字节数,不论变量中存储的数据是否为空。而strlen是一个函数,用于计算一个字符串中的字符数,不包括字符串末尾的空字符'\0'。 另外,sizeof可以用于计算任意数据类型的大小,包括基本数据类型和自定义数据类型。 而strlen只能用于计算字符串的长度。 因此,sizeof通常用于确定数组所占的存储空间,而strlen用于确定字符串的长度。

2、数组和指针

  • 数组 - 能够存放一组相同类型的元素,数组的大小取决于数组的元素个数和元素类型。
  • 指针就是地址,大小都是4/8。(32位机器是4字节,64位机器是8字节)

3、数组名

  • 大部分情况下数组名是首元素地址
  • 2个例外 sizeof(数组名) —— 数组名表示整个数组,计算的是整个数组的大小。(看数组名是否是单独放在sizeof()里面) &数组名 ————-数组名表示整个数组,取出的是数组的地址

二、sizeof()、strlen()的对比

1、注意区别:

  • 字符数组 - char arr1[]={'a','b','c','d','e','f'} ->[a,b,c,d,e,f]
  • 字符串数组 - char arr2[]="abcdef" ->[a,b,c,d,e,f,/0]

  • sizeof()计算的是占用内存空间的大小,单位是字节,不关注内存中到底存放的是什么
  • strlen()是针对字符串的,求的是字符串的长度,本质上统计的是/0之前出现的字符个数。

2、一维数组与一级指针


整型数组

	int a[] = { 1,2,3,4 };
	
	printf("%d\n", sizeof(a[1]));//4
	
	printf("%d\n", sizeof(a));//16 
	//sizeof(a) 就是数组名单独放在sizeof内部 计算的是整个数组的大小。

	printf("%d\n", sizeof(a + 0));// 4/8
	//a + 0   a不是单独放在sizeof()里面,则是数组首元素的地址,sizeof则是计算数组首元素地址的大小
	
	printf("%d\n", sizeof(*a));//4
	//a不是单独放在sizeof()里面,则是数组首元素的地址 *a == a[0],sizeof()则是计算首元素的大小。
	//*a -> *&a[0] -> a[0]

	printf("%d\n", sizeof(a + 1));// 4/8
	//a不是单独放在sizeof()里面,则是数组首元素的地址 -- int*
	//a+1 跳过1个整型,是第二个元素的地址 sizeof()是计算第二个元素地址的大小。

	printf("%d\n", sizeof(&a));/ /4/8
	//&a - 取出的是数组的地址,但是数组的地址也是地址,地址的的大小就是4/8。
	//int (*pa)[4] = &a;

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

	printf("%d\n", sizeof(&a+1));// 4/8
	//&a --> int (*)[4],取出的是数组的地址。
	//&a + 1 则要跳过一个数组。

	printf("%d\n", sizeof(&a[0]));
	//取出首元素地址,sizeof计算的是元素的地址4/8
	
	printf("%d\n", sizeof(&a[0]+1));//4/8
	//取出的是第二个元素的地址,sizeof()计算的是地址的大小。

字符数组

	char arr[] = {'a','b','c','d','e','f'};
	
	printf("%d\n", sizeof(arr));//6
	
	printf("%d\n", sizeof(arr + 0));// 4/8
	//arr不单独放在sizeof里面,是数组首元素的地址 ,计算的大小为4/8
	
	printf("%d\n", sizeof(*arr));//1
	//arr不单独放在sizeof里面,arr是首元素地址,*arr 是首元素,大小为1
	
	printf("%d\n", sizeof(arr[1]));//元素'b'
	
	printf("%d\n", sizeof(&arr));// 4/8
	//&arr 是数组的地址,地址的大小的都是 4/8
	
	printf("%d\n", sizeof(&arr + 1));//4/8
	printf("%d\n", sizeof(&arr[0] + 1));//4/8
	
	//*****************************************
	
	printf("%d\n", strlen(arr));//随机值
	//arr数组首元素地址,找不到‘\0’,所以是计算的大小是随机值
	printf("%d\n", strlen(arr + 0));//随机值,同上
	
	printf("%d\n", strlen(*arr));//非法访问
	//strlen传入char*(地址)
	//*arr = 'a'——>(97)被当成地址,非法访问
	
	printf("%d\n", strlen(arr[1]));//非法访问,同上
	
	printf("%d\n", strlen(&arr));//随机值
	//找不到/0
		
	printf("%d\n", strlen(&arr + 1));//随机值 - 6
	printf("%d\n", strlen(&arr[0] + 0));//随机值 - 1

字符串数组

	char arr[]="abcdef";//[a b c d e f /0]
	
	printf("%d\n",sizeof(arr));//整个数组大小 7
	printf("%d\n",sizeof(arr+0));//首元素地址大小 4/8
	
	printf("%d\n",sizeof(*arr));//首元素大小 1
	//*arr = arr[0] = *(arr+0)
	
	printf("%d\n",sizeof(arr[1]));//1
	
	printf("%d\n",sizeof(&arr));//数组的地址,大小 4/8
	printf("%d\n",sizeof(&arr+1));//+1 跳过一个数组的地址,大小 4/8
	printf("%d\n",sizeof(&arr[0]+1));//+1 跳过一个元素的地址,大小 4/8
	
	//******************************************************************

	printf("%d\n",strlen(arr));//arr表示首元素地址,计算得到数组大小为 6
	printf("%d\n",strlen(arr+0));//6,同上
	
	printf("%d\n",strlen(*arr));//非法访问
	printf("%d\n",strlen(arr[1]));//非法访问
	//strlen是传入char*地址,*arr和arr[1]是数组的元素,传给strlen会被当做地址,造成非法访问
	
	printf("%d\n",strlen(&arr));//&arr表示整个数组的大小,6
	printf("%d\n",strlen(&arr+1));//数组的地址 +1后就找不到 /0 ,大小为随机值
	printf("%d\n",strlen(&arr[0]+1));//从第二个元素开始算,大小为5


指针

	char* p="abcdef";
	printf("%d\n",sizeof(p));//4/8
	//指针变量p 存放着的是地址,所以大小为4/8
	
	printf("%d\n",sizeof(p+1));//字符’b‘的地址,大小为4/8
	
	printf("%d\n",sizeof(*p));
	//解引用的一个char类型元素,大小为1
	
	printf("%d\n",sizeof(p[0]));//1
	//p[0] = *(p+0)
	printf("%d\n",sizeof(&p));//4/8
	
	printf("%d\n",sizeof(&p+1));
	//+1跳过一个char*地址,还是地址,大小还是4/8

	//********************************************

	printf("%d\n",strlen(p));//大小为 6
	printf("%d\n",strlen(p+1));//p+1是'b'的地址,从'b'开始算,大小为5
	printf("%d\n",strlen(*p));//*p为'a',不是地址,非法访问
	
	printf("%d\n",strlen(&p));//随机值
	//&p取的是变量p的地址,不是p里面存放的地址,所以无法找打字符串
	printf("%d\n",strlen(&p+1));//随机值,同上
	
	printf("%d\n",strlen(&p[0]+1));//大小为 5
	//p[0]是’a‘,&p[0]去'a'的地址,+1的'b'的地址,从'b'开始计算
	

3、二维数组与二级指针

整型数组

	int a[3][4]={0};//三行四列的数组
	printf("%d\n",sizeof(a));//4*3*4=48
	//a这个数组名单独放在sizeof里面,表示整个数组,计算的是整个数组的大小
	
	printf("%d\n",sizeof(a[0][0]));//第一行第一个元素,大小4字节
	
	printf("%d\n",sizeof(a[0]));//16
	//a[0] 是第一行的数组名,这是数组名单独放在sizeof内部,计算的是第一行数组的大小
	
	printf("%d\n",sizeof(a[0]+1));//4/8
	//a[0]不是单独放在sizeof内部,表示的是首元素的地址。即第一行第一个元素的地址,相当于&a[0][0]
	//a[0]+1 是第一行第二个元素的地址,相当于&a[0][1]
	
	printf("%d\n",sizeof(a+1));//4/8
	//a作为二维数组的数组名,并非单独放在sizeof内部,所以表示首元素是地址
	//二维数组的首元素是第一行,这里的a就是第一行的地址
	//a+1是跳过第一行,指向第二行
	
	printf("%d\n",sizeof(*(a+1)));//16
	//*(a+1) -> a[1],计算的是第二行的大小
	
	printf("%d\n",sizeof(&a[0]+1));//4/8
	//&a[0]是第一行的地址
	//&a[0]+1是第二行的地址
	
	printf("%d\n",sizeof(*(&a[0]+1)));//表示第二行,大小为16
	printf("%d\n",sizeof(*a));//16
	//*a --> *(a+0)表示第一行,大小为16
	

三、总结回顾

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

标签:arr,地址,数组,printf,sizeof,strlen,指针
From: https://blog.51cto.com/u_16312968/8298002

相关文章

  • 利用快慢指针,求链表的中间结点,判断链表是否是环形链表
    前言(1)在学习数据结构链表部分的时候,老师给出了几个题目。其中两个题目采用了快慢指针的技术,感觉有意思,于是写一篇博客记录一下。快慢指针(1)我们先来介绍一下快慢指针技术。这个说起来其实很简单,就是龟兔赛跑问题。(2)兔子跑的比乌龟快,我们可以利用这个特性,来解决一些实际按理。求链表......
  • 三数之和(双指针法)
    给你一个包含n个整数的数组 nums,判断 nums 中是否存在三个元素a,b,c,使得 a+b+c=0?请你找出所有满足条件且不重复的三元组。注意:答案中不可以包含重复的三元组。示例:给定数组nums=[-1,0,1,2,-1,-4],满足要求的三元组集合为:[[-1,0,1],[-1,-1,2]]......
  • 深入理解sizeof
    最近在论坛里总有人问关于sizeof的问题,并且本人对这个问题也一直没有得到很好的解决,索性今天对它来个较为详细的总结,同时结合strlen进行比较,如果能对大家有点点帮助,这是我最大的欣慰了。一、首先看看sizeof和strlen在MSDN上的定义首先看一MSDN上如何对sizeof进行定义的:sizeofO......
  • C语言程序设计 数组,结构体和指针练习题
    涉及知识点:数组,结构体和指针分析以下程序的运行结果:#include"stdio.h"structsp{inta;int*b;}*p;intd[3]={10,20,30};structspt[3]={70,&d[0],80,&d[1],90,&d[2]};voidmain(){p=t;printf(&......
  • 07-双指针、滑动窗口
    7.双指针、滑动窗口7.1含有全部字符的最短字符串1.题目https://leetcode.cn/problems/minimum-window-substring/给定两个字符串s和t。返回s中包含t的所有字符的最短子字符串。如果s中不存在符合条件的子字符串,则返回空字符串""。如果s中存在多个符合条件的......
  • C语言程序设计 第七章 指针
    本节是学习C语言指针:指针与一般变量,指针与数组,指针与结构体,指向指针的指针。 下载图片格式的课件(PPT课件转换为JPG图片)(以图片方式查看,可以在MP4上查看) 下载Powerpoint课件(在装有PowerPoint的计算机上可以打开使用)......
  • C语言程序设计 练习题参考答案 第七章 (1) 指针与变量 指针与数组
    /*7.13输入三个整数,从小到大排序,(指针,函数实现交换)*/#include"stdio.h"#include"conio.h"voidswap(int*a,int*b,int*c);voidmain(){intx,y,z;printf("请输入三个整数,示例123\n");scanf("%d%d%d",&x,&y,&am......
  • C语言程序设计 第七章 指针与结构体 指针数组 例题
    /*---------------------------------------例7.19输入N个学生学号,姓名,成绩,并按成绩降序排列,并输出p指向结构体变量s1,则s1.成员名,(*p).成员名,p->成员名等价。本题采用自定义函数较为合适Author:emanleeDate:2008-05-12----------------......
  • C语言程序设计 练习题参考答案 第七章 (2) 指针与数组 main函数形参
    /*7.16实现测试字符串长度函数strlen()*/#include"stdio.h"intstrlen(char*p);voidmain(){chars1[20]="s1s2s3s4";char*p=s1;printf("s1的长度:%d\n",strlen(s1));printf("s1的长度:%d\n",strlen(p));......
  • 2008秋-计算机软件基础-结构体与指针复习
    //结构体与指针#include<string.h>#include<stdio.h>structstudent{intnumber;charname[10];};voidmain(){structstudenta;structstudent*ptr=&a;a.number=10;//ptr->number=10;strcpy(a.name,"li");//strcpy(ptr->......