前言:相信很多初学者都对被sizeof和strlen搞得晕晕的,相信看完这篇文章,你对这二者的认识将提升一个档次!已经了解sizeof和strlen的大佬们就当复习吧,哈哈哈~


目录

一.sizeof()详解

二.strlen()详解

strlen()的模拟实现:

法1:计数器

法2:指针-指针

法3:递归

三.strlen与sizeof的区别

四.笔试题

1.整形数组int a[] = {1,2,3,4}

strlen()相关题目

sizeof()相关题目

2.字符数组-char arr[] = {'a','b','c','d','e','f'}

strlen()相关题目

sizeof()相关题目

3.字符数组-char arr[] = “abcdef”

strlen()相关题目

sizeof()相关题目

4.指针指向的常量字符串-char *p = "abcdef"

strlen()相关题目

sizeof()相关题目

5.二维数组-int a[3][4]

sizeof()相关题目

五.总结:


一.sizeof()详解

1.sizeof()作用:计算变量/类型所占内存大小,单位是字节

int a = 10;
int b =sizeof(a);    //a为int类型,大小为4个字节
int c = sizeof(int);   //4

2.sizeof是操作符,不是函数!!! 

sizeof 变量 或者 sizeof(变量)都可以 

sizeof(类型)可以      sizeof 类型是错误的

int main()
{int a, b, c, d;a = sizeof(a);b = sizeof a;c = sizeof(int);//d = sizeof int;    //errprintf("%d %d %d\n", a, b, c);    // 4 4 4return 0;
}

3.sizeof ()内部的表达式不参与真实运算!这点很重要!其运算值在编译时就计算好了 

int main()
{int a = 10;int b = sizeof(a = a + 1);printf("%d %d \n", a, b);    // 10 4return 0;
}

sizeof内部即使写的是赋值表达式也不会去真实的运算,sizeof在计算的时候只看操作数的类型,不会访问对应的空间


4.sizeof()与数组名的关系

sizeof(数组名):此时的数组名代表的是整个数组

&数组名:此时的数组名代表的也是整个数组

其他情况,数组名代表的是首元素地址

int arr[10];
a = sizeof(arr);    //计算的是整个数组的大小 4*10 = 10
b = sizeof(&arr);   //取出整个数组的地址,是地址(指针),4/8

5.sizeof()返回类型造成的坑

int i ;  //全局变量未初始化系统默认为0
int main()
{i--;   //i变成-1if(i > sizeof(i)){printf(">\n");}else{printf("<\n");}return 0;
}

 打印结果: >

注意:sizeof(i)和i比较时,sizeof返回类型为size_t 而i的类型为int,最终是用size_t比较,对于-1的补码为:全1序列,如果看成是无符号数的话,对于的值比sizeof(i) = 4大


二.strlen()

strlen():求字符串长度的库函数,遇到\0即停止计算。需要引用#include<string.h>头文件,注意返回类型为:size_t(unsigned int 无符号整型),

函数原型:


strlen()的模拟实现:

法1:计数器

size_t my_strlen(const char* str)
{size_t count = 0;while (*str != '\0'){count++;str++;}return count;
}

法2:指针-指针

指针-指针得到的是二者之间元素的个数。所以只要一个指针指向字符串首字符,一个指针指向\0,二者相减就是字符串长度

size_t my_strlen2(const char* str)
{char* start = str;char* end = str;while (*end != '\0'){end++;}return end - start;
}

法3:递归

//法3:递归
size_t my_strlen3(const char* str)
{//如果不是\0就+1(本身指向字符),然后递归下一个字符if (*str != '\0')return 1 + my_strlen3(str + 1);//遇到\0即返回0elsereturn 0;
}

strlen()返回参数size_t造成的坑点:

int main()
{char* p1 = "abcd";char* p2 = "abcde";if (strlen(p1) - strlen(p2)>0){printf("p1>p2");}else{printf("p1<p2");}return 0;
}

上述代码打印结果:p1>p2

strlen(p1) 结果为4  strlen(p2):结果为5  二者的类型都为size_t   无符号整形

二者相减得到-1,也被认为是无符号整形,对应的值>0


三.strlen与sizeof的区别

共同点:返回类型都是size_t

不同点:sizeof()是操作符计算的是变量/类型所占空间的大小,单位是字节,\0也算进空间

              strlen()是库函数,计算的是字符串长度,不计算\0

int main()
{char arr1[] = { 'a','b','c' };int ret1 = strlen(arr1);int ret2 = sizeof(arr1);printf("%d %d\n", ret1,ret2);    //随机值    3
//arr1并没放入\0,strlen()向后直到找到\0才停止,所以是随机值  而arr1本身数组元素个数为3,根据后面初始化的内容确定了数组的大小,sizeof:3*1 = 3char arr[] = "abcdef";int ret3 = strlen(arr);int ret4 = sizeof(arr);printf("%d %d\n", ret3, ret4);    //6 7
//arr大小为7,含\0  strlen不算\0return 0;
}

四.笔试题

1.整形数组int a[] = {1,2,3,4}

sizeof()相关题目

int main()
{int a[] = { 1,2,3,4 };printf("%d\n", sizeof(a));//数组名a单独放在sizeof内部,数组名表示整个数组,计算的是整个数组的大小printf("%d\n", sizeof(a + 0));//a表示首元素的地址,a+0还是首元素的地址,地址的大小是4/8字节printf("%d\n", sizeof(*a));   //a表示首元素的地址,*a 就是首元素 ==> a[0] ,大小就是4//*a <==> *(a+0) <==> a[0]printf("%d\n", sizeof(a + 1));//a表示首元素的地址,a+1是第二个元素的地址,大小就是4/8printf("%d\n", sizeof(a[1])); //a[1] 就是第二个元素 - 4printf("%d\n", sizeof(&a));   //&a - 数组的地址 - 4/8 - int(*)[4]printf("%d\n", sizeof(*&a));  //*&a - &a是数组的地址,对数组的地址解引用拿到的是数组,所以大小时候16//相当于printf("%d\n", sizeof(a));//16printf("%d\n", sizeof(&a + 1));//4/8 &a是数组的地址,&a+1 是数组的地址+1,跳过整个数组,虽然跳过了数组,//还是地址  4/8printf("%d\n", sizeof(&a[0]));//4/8printf("%d\n", sizeof(&a[0] + 1));//第二个元素的地址 4/8return 0;
}

2.字符数组-char arr[] = {'a','b','c','d','e','f'}

strlen()相关题目

//字符数组
char arr[] = {'a','b','c','d','e','f'};
//arr中是没有放\0的,而strlen()求长度是找到\0才停止printf("%d\n", strlen(arr));//从arr位置(首元素地址)向后求长度,随机值printf("%d\n", strlen(arr+0));//从arr位置(首元素地址)向后求长度,随机值//printf("%d\n", strlen(*arr));//arr是首元素地址,*arr是字符‘a’-ascii-97,strlen把字符a对应的ascii码值97作为地址向后计数,非法访问!err//printf("%d\n", strlen(arr[1]));//strlen把字符b对应的ascii码值98作为地址向后计数,非法访问!errprintf("%d\n", strlen(&arr));//&arr和arr地址值相同,都是首元素地址,但是意义不一样
//&arr传给strlen  &arr类型:数组指针 char(*p)[6] 而strlen接收的类型为char*,不兼容,但是问题不大
//从数组首元素位置向后计数,随机值printf("%d\n", strlen(&arr+1));//跳过整个数组后,向后计数,随机值-6
//原因:内存空间连续,同时找到\0停止,但是strlen(arr)和strlen(&arr)得到的随机值比第二个多6个字符abcdefprintf("%d\n", strlen(&arr[0]+1));//从字符b位置向后计数,随机数-1

sizeof()相关题目

//字符数组
char arr[] = {'a','b','c','d','e','f'};
//&arr的类型:数组指针: char(*)[6]printf("%d\n", sizeof(arr));//数组名单独放在sizeof内部,计算的是整个数组的大小,元素个数为6个(不含\0),类型为char 所以大小为6printf("%d\n", sizeof(arr+0));//此处的arr代表的是首元素地址,arr+0仍是首元素地址char*,地址(指针)大小是4/8printf("%d\n", sizeof(*arr));//此处的arr代表的是首元素地址,*arr即为数组首元素,即为字符‘a’  大小为1printf("%d\n", sizeof(arr[1]));//arr[1]->字符‘b’,大小为1printf("%d\n", sizeof(&arr));//取出整个数组的地址,还是地址,地址的大小就是4/8  printf("%d\n", sizeof(&arr+1));//取出数组arr的地址+1,跳过一个数组,还是地址,地址的大小为:4/8printf("%d\n", sizeof(&arr[0]+1));//数组第二个元素的地址,4/8

3.字符数组-char arr[] = “abcdef”

strlen()相关题目

char arr[] = "abcdef";
//此时数组arr中存放了\0  strlen求长度,遇到\0即停止计数printf("%d\n", strlen(arr));//从arr位置开始向后计数,遇到\0即停,长度为6printf("%d\n", strlen(arr+0));///从arr位置开始向后计数,遇到\0即停,长度为6//printf("%d\n", strlen(*arr));//arr是首元素地址,*arr是字符‘a’对应ascii值为97,strlen把字符a对应的ascii码值97作为地址向后计数,非法访问!err//printf("%d\n", strlen(arr[1]));///arr[1]:‘b’对应ascii值为98,strlen把字符b对应的ascii码值98作为地址向后计数,非法访问!errprintf("%d\n", strlen(&arr));&arr和arr地址值相同,都是首元素地址,但是意义不一样
//&arr传给strlen  &arr类型:数组指针 char(*p)[6] 而strlen接收的类型为char*,不兼容,但是问题不大
//从数组首元素位置向后计数,值为 6printf("%d\n", strlen(&arr+1));//跳过整个数组后,向后计数,未知值printf("%d\n", strlen(&arr[0]+1));//从b未知向后计数,长度为5

sizeof()相关题目

char arr[] = "abcdef";
//此时的arr数组里面是放了\0的printf("%d\n", sizeof(arr));//数组名单独放在sizeof内部,计算的是整个数组的大小,\0也算进去,大小为7printf("%d\n", sizeof(arr+0));//此时的数组名是首元素地址,地址(指针)大小:4/8printf("%d\n", sizeof(*arr));//此时的数组名是首元素地址,*arr即为首元素,字符a->char类型,大小为1printf("%d\n", sizeof(arr[1]));//arr[1]:字符'b',大小为1printf("%d\n", sizeof(&arr));//取出数组的地址,还是地址,大小为4/8printf("%d\n", sizeof(&arr+1));//取出数组的地址+1,跳过整个数组,还是地址:4/8printf("%d\n", sizeof(&arr[0]+1));//取出第一个元素的地址+1,跳过一个元素,即为第二个元素的地址,4/8

4.指针指向的常量字符串-char *p = "abcdef"

strlen()相关题目

const char* p = "abcdef";
//p存放的是字符a的地址,
//p+1:字符b的地址printf("%d\n", strlen(p));//p存放的是字符a的地址,即从字符a的地址向后计数,长度为6printf("%d\n", strlen(p+1));//从字符b的地址向后计数,长度为5//printf("%d\n", strlen(*p));//*p ->字符‘a’  即以字符a的ascii码值97为地址向后计数,非法访问,err//printf("%d\n", strlen(p[0]));//p[0] ->字符‘a’  即以字符a的ascii码值97为地址向后计数,非法访问,errprintf("%d\n", strlen(&p));//&p取出的是p变量的地址,即以p变量的地址(16进制)向后计数,  随机值printf("%d\n", strlen(&p+1));//&p取出的是p变量的地址,&p+1,跳过p变量,即从p变量之后的位置向后访问  随机值printf("%d\n", strlen(&p[0]+1));//&p[0]==>相当于&*(p+0)-->相当于sizeof(p),p存中存放的是字符a的地址,+1,即为字符b的地址,从字符b位置向后访问,  长度为5
// p[0] :字符a
//&p[0]:字符a的地址
//&p[0] +1:字符b的地址//注意上面两个随机值没有必然关系,因为strlen(&p)是以p地址对应的16进制向后访问,值为未知数,有可能提前遇到\0结束了

sizeof()相关题目

//因为指针指向的是常量字符串,不可以被修改
//所以可以用const修饰
//char* p = "abcdef";
const char * p ="abcdef";
//p存放的是字符a的地址printf("%d\n", sizeof(p));//p是指针,指向字符a,大小为4/8printf("%d\n", sizeof(p+1));//p+1,指向的是字符b,指针,大小为4/8printf("%d\n", sizeof(*p));//p存放的是字符a的地址,*p:即为字符a,大小为1printf("%d\n", sizeof(p[0]));//p[0]->字符a ,大小为1printf("%d\n", sizeof(&p));//取出p变量的地址,仍是地址,大小为4/8printf("%d\n", sizeof(&p+1));//取出p变量的地址+1,跳过p变量,但仍是地址,大小为4/8printf("%d\n", sizeof(&p[0]+1));//&p[0]相当于&*(p+0),&和*抵消,&p[0]:字符a的地址,+1:字符b的地址,大小为4/8

5.二维数组-int a[3][4]

内存布局

实际内存结构

想象结构

sizeof()相关题目

//二维数组
int a[3][4] = {0};printf("%d\n",sizeof(a));//数组名单独放在sizeof内部,计算的是整个数组的大小, 数组元素为12个,每一个元素大小为4个字节,12*4=48printf("%d\n",sizeof(a[0][0]));//计算的是数组第有一行第一个元素的大小,int类型,大小为4printf("%d\n",sizeof(a[0]));//a[0]==>*(a+0)==>数组名是首元素地址,即为第一行的地址,解引用第一行的地址,就是第一行,所以计算的是第一行元素的大小 4*4=16
//a[0] : 二维数组的第一行printf("%d\n",sizeof(a[0]+1));//a[0]:第一行的数组名,代表第一行第一个元素的地址,a[0]+1::跳过一个元素,即为第一行第二个元素地址,大小为4/8
//注意:a[0] + 1 :不是第二行,a[0]是第一行的数组名,首元素地址,即为第一行第一个元素地址,a[0]+1:跳过一个元素   a+1:a为数组名,首元素地址,第一行的地址,a+1,跳过一行,二维数组第二行printf("%d\n",sizeof(*(a[0]+1)));//由上可得:a[0]+1:第一行第二个元素地址, *(a[0]+1):即为第一行第二个元素 int类型 大小为4printf("%d\n",sizeof(a+1));//a为二维数组的数组名->首元素地址,e二维数组第一行的地址,+1,跳过一行,即为第二行的地址->地址,大小为4/8printf("%d\n",sizeof(*(a+1)));//由上,a+1是第二行的地址,*(a+1)即为第二行,大小为4*4 = 16printf("%d\n",sizeof(&a[0]+1));//a[0]是第一行的数组名,&a[0]就是第一行的地址,(相当于是,数组名和取地址数组名的关系,二二者地址值相同,但是含义不同),&a[0]+1:跳过第一行,即为第二行地址,地址:4/8printf("%d\n",sizeof(*(&a[0]+1)));//由上:(&a[0]+1):第二行地址,解引用就是第二行,大小为4*4 = 16printf("%d\n",sizeof(*a));//二维数组数组名是首元素地址,即为第一行的地址,解引用就是第一行, 大小为4* 4  = 16printf("%d\n",sizeof(a[3]));//a[3]假设存在,就是第四行的数组名,sizeof(a[3])相当于数组名单独放在sizeof内部,计算的是第四行的大小, 4*4 = 16//sizeof内部的表达式不参与运算,即不会真的去访问a[3]的空间,所以不出错,它只看一下第四行的类型,并没有真正去访问第四行的内容,

五.总结:

strlen()和sizeof()坑点挺多的,大家认真学习!一起加油哈哈哈!

如果感觉对你有帮助的,欢迎点个赞评论一下呀!


C语言sizeof与strlen详解(附大量笔试题题解过程)相关推荐

  1. C语言sizeof与strlen详解

    sizeof()详解 1.sizeof()作用:计算变量/类型所占内存大小,单位是字节.(计算字符串函数大小时包含'\0') int a = 10; int b =sizeof(a); //a为int ...

  2. 百度面试过程详解-附电话面试题

    经过在线笔试.两轮电话面试,今天上午收到了百度的拒信,我的百度求职算是告一段落了-- 从百度校园招聘开始,我就投了一份简历.在别人都有在线笔试机会的时候,我却没有任何消息. 听说师兄可以给推荐,我就又 ...

  3. sizeof 在C语言的作用,C语言中的sizeof的用法详解

    C语言中的sizeof的用法详解 一.sizeof的概念 sizeof是C语言的一种单目操作符,如C语言的其他操作符++.--等.它并不是函数.sizeof操作符以字节形式给出了其操作数的存储大小.操 ...

  4. C语言qsort快速排序函数详解

    直接进入主题,在c语言中qsort函数是用来快速排序的,qsort有4个参数,分别是数组地址,数组元素个数,数组元素字节大小和一个比较数组元素的函数指针.让我来看一下官方给出的使用标准,上图: 让我们 ...

  5. C语言再学习 -- 详解C++/C 面试题 2

    (经典)C语言测试:想成为嵌入式程序员应知道的0x10个基本问题. 参看:嵌入式程序员面试问题集锦 1.用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽略闰年问题) #define ...

  6. 用数据结构c语言写成绩排序,C语言数据结构 快速排序实例详解

    C语言数据结构 快速排序实例详解 一.快速排序简介 快速排序采用分治的思想,第一趟先将一串数字分为两部分,第一部分的数值都比第二部分要小,然后按照这种方法,依次对两边的数据进行排序. 二.代码实现 # ...

  7. c语言中快排函数,c语言快排函数详解

    c语言快排函数详解 int cmp(const void *a, const void *b) 返回正数就是说 cmp 传入参数第一个要放在第二个后面, 负数就是传入参数第一个要放第二个前面, 如果是 ...

  8. java文档注释定界符_c语言的注释定界符详解

    c语言的注释定界符详解 c语言的注释定界符是什么 1.最早期的C语言注释是:/* */ 2.后来又增加的行注释:// 其中/**/是多行注释,//是单行注释. 需要注意的是:C 语言的注释并不是可以出 ...

  9. 【C】C语言格式输入函数scanf()详解

    参考了:C语言格式输入函数scanf()详解 总述 scanf函数称为格式输入函数,即按用户指定的格式从键盘上把数据输入到指定的变量之中. scanf函数的一般形式 scanf函数是一个标准库函数,它 ...

  10. C语言return的用法详解,C语言函数返回值详解。 (本次转载仅供学习,感谢原创!!转发自C语言中文网,如有侵权请私信本人删除)

    C语言return的用法详解,C语言函数返回值详解 转载:http://c.biancheng.net/view/1855.html 函数的返回值是指函数被调用之后,执行函数体中的代码所得到的结果,这 ...

最新文章

  1. pyhton re模块
  2. 64位游戏找call_替换Unity可执行文件为64位,改善游戏性能
  3. 清华大学史作强副教授专访:用流形、偏微分方程揭秘人工智能
  4. 科学计算库Numpy——随机模块
  5. GIS输出PDF为什么标注有问题
  6. 游戏开发--开源软件11--Firefly(python 服务端分布式框架)||pygame
  7. [每天一道A+B]签到检测程序
  8. JavaScript实现RadixSort基数排序算法(附完整源码)
  9. Mirror--镜像使用的工作线程数
  10. 书籍折页是什么效果_Word的书籍折页是什么
  11. 关于spring注入
  12. 关于虚拟机xp系统上不了网的问题
  13. 如何修改apk文件,改之理简单使用教程
  14. Windows 下利用cWrsync同步
  15. 如何注册PayPal账户
  16. eplan窗口宏与符号宏是什么_电气设计||Eplan P8 宏功能的应用
  17. Linux内核框架之内核进程
  18. AI遮天传 ML-初识决策树
  19. Profibus网络故障诊断技术总结
  20. Firefox的about:config设置详解

热门文章

  1. Linux 人大金仓安装部署记录
  2. 一个SAP开发人员的双截棍之路
  3. 个人网站与个人博客的区别
  4. 使用Chrome开发者工具精确定位网页元素位置
  5. android 视频截屏代码,android视频截屏手机录屏实现代码
  6. iOS中什么是superView?(新手概念简述版)
  7. 张云茹计算机,张云茹 - 重庆理工大学 - 药学与生物工程学院
  8. 价值博客们,技术博客
  9. java调用存储过程 sql server,Sql Server的存储过程与Java代码相连接调用(二)
  10. Word 之 清除页眉下划线