C语言基础——sizeof()和strlen()傻傻分不清怎么破?

在平时的C语言学习中,有很多朋友可能对sizeof()和strlen()的用法混淆不清,今天lucky同学以一维数组的访问为例,来对两者的用法加以介绍和区分。

sizeof()的用法

sizeof不是函数,是一个操作符,sizeof计算的是对象所占内存的大小,单位是字节,它不在乎内存中存放的是什么,只在乎内存的大小。

对于数组的访问,首先要清楚这几个概念:
数组名通常为首元素的地址,但有两个例外:
1、sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小
2、&数组名,这里的数组名表示整个数组,取出的是整个数组的地址(数组的地址也是地址,在32位平台下占4个字节,在64位平台下占8个字节)

搞清楚上面的概念后,来看如下几个例子:

#include <stdio.h>
int main()
{
    int a[] = {1, 2, 3, 4};

    printf("%d\n",sizeof(a));    // 16 
    //sizeof(a),此时的数组名表示整个数组的大小,四个整型即16字节 
 
    printf("%d\n",sizeof(a+0));    // 4/8
    //sizeof里放的不是单独的a,则a代表首元素的地址,(a+0)也是数组首元素的地址,所以为4或8字节

    printf("%d\n",sizeof(*a));    // 4
    //a表示数组首元素的地址,*a表示数组的第一个元素,sizeof(*a)就是第一个元素的大小

    printf("%d\n",sizeof(a+1));    // 4/8
    //a表示数组首元素的地址,a+1是第二个元素的地址,sizeof(a+1)就是第二个元素的地址的大小

    printf("%d\n",sizeof(a[1]));    // 4
    //表示数组的第二个元素的大小

    printf("%d\n",sizeof(&a));    // 4/8
    //&a取出的是首元素地址的地址,也是地址,大小就是4或8个字节

    printf("%d\n",sizeof(*&a));    // 16
    //&a取得是整个数组大小,再*就拿到整个数组,计算的是整个数组的大小

    printf("%d\n",sizeof(&a+1));    // 4/8
    &a拿到的是整个数组的地址,加1跳过整个数组,产生的是4后面位置的地址,还是一个地址

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

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

    return 0;
}

以上是整形数组,再来看字符数组:

#include <stdio.h>
int main()
{
	char arr[] = {'a','b','c','d','e','f'};    // [a b c d e f]

	printf("%d\n", sizeof(arr));    // 6
    //sizeof(arr),这里的数组名表示整个字符数组,一个字符为1个字节

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

	printf("%d\n", sizeof(*arr));    // 1
    //*arr指的是数组的首元素'a',为一个字节

	printf("%d\n", sizeof(arr[1]));    // 1
    //arr[1]表示数组第二个元素'b',为一个字节

	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
    //(&arr[0]+1) 表示数组第二个元素的地址

return 0;
}

 strlen()的用法

strlen () 是一个库函数,主要是用来求字符串长度(内存里面只能放字符),从给定的地址一直向后访问字符,直到遇到 \0 为止,strlen() 统计的是 \0 之前字符的个数。strlen () 函数的定义: size_t  strlen(const  char*  string);可以看到 strlen() 函数里面放的是一个const char*类型的指针变量,所以 () 里面默认是一个地址。

了解清楚 strlen() 的概念,再来看以下例子来深入理解:

#include <stdio.h>
int main()
{
    char arr[] = {'a','b','c','d','e','f'};    // [a b c d e f]

    printf("%d\n", strlen(arr));    // 随机值 
    //arr表首元素地址,即从首元素开始数,arr数组中没有\0,所以strlen函数会继续往后找\0,统计\0之前出现的字符个数,由于不知道何时才会找到\0,所以为随机值

	printf("%d\n", strlen(arr+0));    // 随机值
    //和上面一样,(arr+0) 还是第一个元素的地址,仍然是从第一个元素开始数,strlen函数会继续往后找\0,统计\0之前出现的字符个数

	printf("%d\n", strlen(*arr));    //err 非法访问
    //这里指的是第一个元素字符a,ASCII为97,strlen函数从97的地址开始数,访问了错误的地址,出现了野指针,属于非法访问

	printf("%d\n", strlen(arr[1]));    //err 非法访问
    //同上面一样,这里指的是第二个元素字符b,仍然属于非法访问

	printf("%d\n", strlen(&arr));    // 随机值
    //&arr取得是整个数组的地址,仍然是从数组首元素地址开始,向后找遇到\0之前字符的个数,所以为随机值

	printf("%d\n", strlen(&arr+1));    // 随机值
    //(&arr+1)是从数组地址开始向后跳过了整个数组而产生的一个地址,从这个地址开始,向后找遇到\0之前字符的个数,所以仍是随机值

	printf("%d\n", strlen(&arr[0]+1));    // 随机值
    //(&arr[0]+1) 指的是数组第二个元素的地址,从这个地址向后找\0之前字符的个数,不难知道仍是随机值

return 0;
}

到这里,大家对两者的用法有了更加深入的了解了吗,get it !

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值