关于对 sizeof 和 strlen 函数的应用

本文通过多个例题详细解释了C语言中sizeof运算符和strlen函数在处理数组、指针及地址时的行为差异。sizeof计算内存占用大小,而strlen计算字符串长度。在不同环境(X64/X86)下,结果可能因数据结构对齐规则而异。错误使用这些操作可能导致程序异常。
摘要由CSDN通过智能技术生成
  1. 例题

例题1:

讲解:

printf("%d\n",sizeof(a)); 4*4=16(字节) (1个整形4个字节,a数组中每个元素是整形)

printf("%d\n", sizeof(a+0)); 4/8 (a不是单独出现在sizeof内部因此a表示数组首元素的地址,地址加0不变)

printf("%d\n", sizeof(*a)); 4 (*a表示对数组首元素地址解引用,得到的是一个元素,一个元素为4个字节)

printf("%d\n", sizeof(a+1)); 4/8 ( a为数组首元素的地址+1找到第2个元素的地址)

printf("%d\n", sizeof(a[1])); 4 (计算数组第2个元素的大小)

printf("%d\n", sizeof(&a)); 4/8 ( &a找到的是整个数组的地址)

printf("%d\n", sizeof(*&a)); 16 (计算整个数组的大小)

printf("%d\n", sizeof(&a+1)); 4/8 (&a得到整个数组的地址,+1后跳过整个数组,但本质上还是地址,对应4后面的地址)

printf("%d\n", sizeof(&a[0])); 4/8 (第一个元素的地址)

printf("%d\n", sizeof(&a[0]+1)); 4/8 (数组第2个元素的地址)

实际运行:

(X64环境下)

(X86环境下)

例题2:

讲解:

printf("%d\n", sizeof(arr)); 6 (计算整个数组的大小)

printf("%d\n", sizeof(arr + 0)); 4/8 (arr+0计算数组首元素地址)

printf("%d\n", sizeof(*arr)); 1 (对数组首元素进行解引用)

printf("%d\n", sizeof(arr[1])); 1 (数组第2个元素的大小)

printf("%d\n", sizeof(&arr)); 4/8 (整个数组的地址)

printf("%d\n", sizeof(&arr + 1)); 4/8 (整个数组的地址加1跳到字符f后,但本质上还是地址)

printf("%d\n", sizeof(&arr[0] + 1)); 4/8 (第2个元素的地址计算sizeof)

printf("%d\n", strlen(arr)); 随机值(因为arr数组中无\0,strlen函数则会向后继续找\0,此时在找到\0之前到底有几个字符是未知的)

printf("%d\n", strlen(arr + 0)); 随机值

printf("%d\n", strlen(*arr)); err (strlen函数接收的是地址而此时是对数组首元素解引用找到了字符'a',然后把'a'传了进去,此时把'a'的ASCII码值认为是地址去接收,此时以这个地址去访问就会出错)

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

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

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

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

实际运行:

(X64环境下)

(X86环境下)

(err情况)

(如上图所示,当运行时,系统就报错了)

例题3:

讲解:

printf("%d\n", sizeof(arr)); 7 (计算整个数组的大小)

printf("%d\n", sizeof(arr + 0)); 4/8 (arr+0得到是数组首元素的地址)

printf("%d\n", sizeof(*arr)); 1 (用sizeof计算数组首元素的大小)

printf("%d\n", sizeof(arr[1])); 1 (arr[1]得到的是数组中下标为1的元素并且去计算大小)

printf("%d\n", sizeof(&arr)); 4/8 (&arr得到的是整个数组的地址)

printf("%d\n", sizeof(&arr + 1)); 4/8 (&arr+1得到的是f之后的地址)

printf("%d\n", sizeof(&arr[0] + 1)); 4/8 (&arr[0]得到的是第1个元素的地址,&arr[0]+1得到的是第2个元素的地址)

printf("%d\n", strlen(arr)); 6

printf("%d\n", strlen(arr + 0)); 6

printf("%d\n", strlen(*arr)); err (*arr得到的是数组的首元素)

printf("%d\n", strlen(arr[1])); err (arr[1]得到的是数组的第2个元素)

printf("%d\n", strlen(&arr)); 6

printf("%d\n", strlen(&arr + 1)); 随机值 (&arr得到的是整个数组的地址,&arr+1跳过整个数组得到\0之后的地址并以这个地址向后找\0,得到的是一个随机值)

printf("%d\n", strlen(&arr[0] + 1)); 5 (&arr[0]+1得到的是第2个元素的地址且以这个地址为中心去向后找到\0停止,然后计算中间的字符个数)

实际运行:

(X64环境下)

(X86环境下)

(err情况)

(上面就为报错的情况)

例题4:

讲解:

printf("%d\n", sizeof(p)); 4/8 (p是指针变量,计算的是指针变量的大小)

printf("%d\n", sizeof(p + 1)); 4/8 (p中存储的是字符串中字符a的地址,p+1得到的是b的地址)

printf("%d\n", sizeof(*p)); 1 (p中存的是字符a的地址,对字符a的地址解引用得到的是字符a)

printf("%d\n", sizeof(p[0])); 1 (p[0] -> *(p+0) ->*p,计算字符a)

printf("%d\n", sizeof(&p)); 4/8 (指针变量p在内存中的地址)

printf("%d\n", sizeof(&p + 1)); 4/8 (&p+1是跳过p之后的地址)

printf("%d\n", sizeof(&p[0] + 1)); 4/8 (&p[0]得到的是字符a的地址,&p[0]+1得到的是字符b的地址)

printf("%d\n", strlen(p)); 6

printf("%d\n", strlen(p + 1)); 5

printf("%d\n", strlen(*p)); err

printf("%d\n", strlen(p[0])); err

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

printf("%d\n", strlen(&p + 1)); 随机值

printf("%d\n", strlen(&p[0] + 1)); 5

实际运行:

(X64环境下)

(X86环境下)

(err情况)

  1. 总结

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

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

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

  1. sizeof是一个操作符,sizeof计算的是对象所占内存的大小,单位是字节,返回类型是size_t,它不在乎内存中存放的是什么,只在乎内存的大小。

  1. strlen是一个库函数,是求字符串长度,从给定的地址向后访问字符,统计‘\0'之前出现的字符个数。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值